VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMAll/IEMAllInstructionsTwoByte0f.cpp.h@ 67078

Last change on this file since 67078 was 67034, checked in by vboxsync, 8 years ago

IEM: Tests+docs for movntdq Mdq,Vdq (0x66 0x0f 0xe7).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 331.2 KB
Line 
1/* $Id: IEMAllInstructionsTwoByte0f.cpp.h 67034 2017-05-23 11:10:57Z vboxsync $ */
2/** @file
3 * IEM - Instruction Decoding and Emulation.
4 *
5 * @remarks IEMAllInstructionsVexMap1.cpp.h is a VEX mirror of this file.
6 * Any update here is likely needed in that file too.
7 */
8
9/*
10 * Copyright (C) 2011-2017 Oracle Corporation
11 *
12 * This file is part of VirtualBox Open Source Edition (OSE), as
13 * available from http://www.virtualbox.org. This file is free software;
14 * you can redistribute it and/or modify it under the terms of the GNU
15 * General Public License (GPL) as published by the Free Software
16 * Foundation, in version 2 as it comes in the "COPYING" file of the
17 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19 */
20
21
22/** @name Two byte opcodes (first byte 0x0f).
23 *
24 * @{
25 */
26
27/** Opcode 0x0f 0x00 /0. */
28FNIEMOPRM_DEF(iemOp_Grp6_sldt)
29{
30 IEMOP_MNEMONIC(sldt, "sldt Rv/Mw");
31 IEMOP_HLP_MIN_286();
32 IEMOP_HLP_NO_REAL_OR_V86_MODE();
33
34 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
35 {
36 IEMOP_HLP_DECODED_NL_1(OP_SLDT, IEMOPFORM_M_REG, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
37 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_LDTR_READS, SVM_EXIT_LDTR_READ, 0, 0);
38 switch (pVCpu->iem.s.enmEffOpSize)
39 {
40 case IEMMODE_16BIT:
41 IEM_MC_BEGIN(0, 1);
42 IEM_MC_LOCAL(uint16_t, u16Ldtr);
43 IEM_MC_FETCH_LDTR_U16(u16Ldtr);
44 IEM_MC_STORE_GREG_U16((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u16Ldtr);
45 IEM_MC_ADVANCE_RIP();
46 IEM_MC_END();
47 break;
48
49 case IEMMODE_32BIT:
50 IEM_MC_BEGIN(0, 1);
51 IEM_MC_LOCAL(uint32_t, u32Ldtr);
52 IEM_MC_FETCH_LDTR_U32(u32Ldtr);
53 IEM_MC_STORE_GREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u32Ldtr);
54 IEM_MC_ADVANCE_RIP();
55 IEM_MC_END();
56 break;
57
58 case IEMMODE_64BIT:
59 IEM_MC_BEGIN(0, 1);
60 IEM_MC_LOCAL(uint64_t, u64Ldtr);
61 IEM_MC_FETCH_LDTR_U64(u64Ldtr);
62 IEM_MC_STORE_GREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u64Ldtr);
63 IEM_MC_ADVANCE_RIP();
64 IEM_MC_END();
65 break;
66
67 IEM_NOT_REACHED_DEFAULT_CASE_RET();
68 }
69 }
70 else
71 {
72 IEM_MC_BEGIN(0, 2);
73 IEM_MC_LOCAL(uint16_t, u16Ldtr);
74 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
75 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
76 IEMOP_HLP_DECODED_NL_1(OP_SLDT, IEMOPFORM_M_MEM, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
77 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_LDTR_READS, SVM_EXIT_LDTR_READ, 0, 0);
78 IEM_MC_FETCH_LDTR_U16(u16Ldtr);
79 IEM_MC_STORE_MEM_U16(pVCpu->iem.s.iEffSeg, GCPtrEffDst, u16Ldtr);
80 IEM_MC_ADVANCE_RIP();
81 IEM_MC_END();
82 }
83 return VINF_SUCCESS;
84}
85
86
87/** Opcode 0x0f 0x00 /1. */
88FNIEMOPRM_DEF(iemOp_Grp6_str)
89{
90 IEMOP_MNEMONIC(str, "str Rv/Mw");
91 IEMOP_HLP_MIN_286();
92 IEMOP_HLP_NO_REAL_OR_V86_MODE();
93
94 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
95 {
96 IEMOP_HLP_DECODED_NL_1(OP_STR, IEMOPFORM_M_REG, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
97 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_TR_READS, SVM_EXIT_TR_READ, 0, 0);
98 switch (pVCpu->iem.s.enmEffOpSize)
99 {
100 case IEMMODE_16BIT:
101 IEM_MC_BEGIN(0, 1);
102 IEM_MC_LOCAL(uint16_t, u16Tr);
103 IEM_MC_FETCH_TR_U16(u16Tr);
104 IEM_MC_STORE_GREG_U16((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u16Tr);
105 IEM_MC_ADVANCE_RIP();
106 IEM_MC_END();
107 break;
108
109 case IEMMODE_32BIT:
110 IEM_MC_BEGIN(0, 1);
111 IEM_MC_LOCAL(uint32_t, u32Tr);
112 IEM_MC_FETCH_TR_U32(u32Tr);
113 IEM_MC_STORE_GREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u32Tr);
114 IEM_MC_ADVANCE_RIP();
115 IEM_MC_END();
116 break;
117
118 case IEMMODE_64BIT:
119 IEM_MC_BEGIN(0, 1);
120 IEM_MC_LOCAL(uint64_t, u64Tr);
121 IEM_MC_FETCH_TR_U64(u64Tr);
122 IEM_MC_STORE_GREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u64Tr);
123 IEM_MC_ADVANCE_RIP();
124 IEM_MC_END();
125 break;
126
127 IEM_NOT_REACHED_DEFAULT_CASE_RET();
128 }
129 }
130 else
131 {
132 IEM_MC_BEGIN(0, 2);
133 IEM_MC_LOCAL(uint16_t, u16Tr);
134 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
135 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
136 IEMOP_HLP_DECODED_NL_1(OP_STR, IEMOPFORM_M_MEM, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
137 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_TR_READS, SVM_EXIT_TR_READ, 0, 0);
138 IEM_MC_FETCH_TR_U16(u16Tr);
139 IEM_MC_STORE_MEM_U16(pVCpu->iem.s.iEffSeg, GCPtrEffDst, u16Tr);
140 IEM_MC_ADVANCE_RIP();
141 IEM_MC_END();
142 }
143 return VINF_SUCCESS;
144}
145
146
147/** Opcode 0x0f 0x00 /2. */
148FNIEMOPRM_DEF(iemOp_Grp6_lldt)
149{
150 IEMOP_MNEMONIC(lldt, "lldt Ew");
151 IEMOP_HLP_MIN_286();
152 IEMOP_HLP_NO_REAL_OR_V86_MODE();
153
154 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
155 {
156 IEMOP_HLP_DECODED_NL_1(OP_LLDT, IEMOPFORM_M_REG, OP_PARM_Ew, DISOPTYPE_DANGEROUS);
157 IEM_MC_BEGIN(1, 0);
158 IEM_MC_ARG(uint16_t, u16Sel, 0);
159 IEM_MC_FETCH_GREG_U16(u16Sel, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
160 IEM_MC_CALL_CIMPL_1(iemCImpl_lldt, u16Sel);
161 IEM_MC_END();
162 }
163 else
164 {
165 IEM_MC_BEGIN(1, 1);
166 IEM_MC_ARG(uint16_t, u16Sel, 0);
167 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
168 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
169 IEMOP_HLP_DECODED_NL_1(OP_LLDT, IEMOPFORM_M_MEM, OP_PARM_Ew, DISOPTYPE_DANGEROUS);
170 IEM_MC_RAISE_GP0_IF_CPL_NOT_ZERO(); /** @todo test order */
171 IEM_MC_FETCH_MEM_U16(u16Sel, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
172 IEM_MC_CALL_CIMPL_1(iemCImpl_lldt, u16Sel);
173 IEM_MC_END();
174 }
175 return VINF_SUCCESS;
176}
177
178
179/** Opcode 0x0f 0x00 /3. */
180FNIEMOPRM_DEF(iemOp_Grp6_ltr)
181{
182 IEMOP_MNEMONIC(ltr, "ltr Ew");
183 IEMOP_HLP_MIN_286();
184 IEMOP_HLP_NO_REAL_OR_V86_MODE();
185
186 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
187 {
188 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
189 IEM_MC_BEGIN(1, 0);
190 IEM_MC_ARG(uint16_t, u16Sel, 0);
191 IEM_MC_FETCH_GREG_U16(u16Sel, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
192 IEM_MC_CALL_CIMPL_1(iemCImpl_ltr, u16Sel);
193 IEM_MC_END();
194 }
195 else
196 {
197 IEM_MC_BEGIN(1, 1);
198 IEM_MC_ARG(uint16_t, u16Sel, 0);
199 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
200 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
201 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
202 IEM_MC_RAISE_GP0_IF_CPL_NOT_ZERO(); /** @todo test ordre */
203 IEM_MC_FETCH_MEM_U16(u16Sel, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
204 IEM_MC_CALL_CIMPL_1(iemCImpl_ltr, u16Sel);
205 IEM_MC_END();
206 }
207 return VINF_SUCCESS;
208}
209
210
211/** Opcode 0x0f 0x00 /3. */
212FNIEMOP_DEF_2(iemOpCommonGrp6VerX, uint8_t, bRm, bool, fWrite)
213{
214 IEMOP_HLP_MIN_286();
215 IEMOP_HLP_NO_REAL_OR_V86_MODE();
216
217 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
218 {
219 IEMOP_HLP_DECODED_NL_1(fWrite ? OP_VERW : OP_VERR, IEMOPFORM_M_MEM, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
220 IEM_MC_BEGIN(2, 0);
221 IEM_MC_ARG(uint16_t, u16Sel, 0);
222 IEM_MC_ARG_CONST(bool, fWriteArg, fWrite, 1);
223 IEM_MC_FETCH_GREG_U16(u16Sel, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
224 IEM_MC_CALL_CIMPL_2(iemCImpl_VerX, u16Sel, fWriteArg);
225 IEM_MC_END();
226 }
227 else
228 {
229 IEM_MC_BEGIN(2, 1);
230 IEM_MC_ARG(uint16_t, u16Sel, 0);
231 IEM_MC_ARG_CONST(bool, fWriteArg, fWrite, 1);
232 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
233 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
234 IEMOP_HLP_DECODED_NL_1(fWrite ? OP_VERW : OP_VERR, IEMOPFORM_M_MEM, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
235 IEM_MC_FETCH_MEM_U16(u16Sel, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
236 IEM_MC_CALL_CIMPL_2(iemCImpl_VerX, u16Sel, fWriteArg);
237 IEM_MC_END();
238 }
239 return VINF_SUCCESS;
240}
241
242
243/** Opcode 0x0f 0x00 /4. */
244FNIEMOPRM_DEF(iemOp_Grp6_verr)
245{
246 IEMOP_MNEMONIC(verr, "verr Ew");
247 IEMOP_HLP_MIN_286();
248 return FNIEMOP_CALL_2(iemOpCommonGrp6VerX, bRm, false);
249}
250
251
252/** Opcode 0x0f 0x00 /5. */
253FNIEMOPRM_DEF(iemOp_Grp6_verw)
254{
255 IEMOP_MNEMONIC(verw, "verw Ew");
256 IEMOP_HLP_MIN_286();
257 return FNIEMOP_CALL_2(iemOpCommonGrp6VerX, bRm, true);
258}
259
260
261/**
262 * Group 6 jump table.
263 */
264IEM_STATIC const PFNIEMOPRM g_apfnGroup6[8] =
265{
266 iemOp_Grp6_sldt,
267 iemOp_Grp6_str,
268 iemOp_Grp6_lldt,
269 iemOp_Grp6_ltr,
270 iemOp_Grp6_verr,
271 iemOp_Grp6_verw,
272 iemOp_InvalidWithRM,
273 iemOp_InvalidWithRM
274};
275
276/** Opcode 0x0f 0x00. */
277FNIEMOP_DEF(iemOp_Grp6)
278{
279 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
280 return FNIEMOP_CALL_1(g_apfnGroup6[(bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK], bRm);
281}
282
283
284/** Opcode 0x0f 0x01 /0. */
285FNIEMOP_DEF_1(iemOp_Grp7_sgdt, uint8_t, bRm)
286{
287 IEMOP_MNEMONIC(sgdt, "sgdt Ms");
288 IEMOP_HLP_MIN_286();
289 IEMOP_HLP_64BIT_OP_SIZE();
290 IEM_MC_BEGIN(2, 1);
291 IEM_MC_ARG(uint8_t, iEffSeg, 0);
292 IEM_MC_ARG(RTGCPTR, GCPtrEffSrc, 1);
293 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
294 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
295 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
296 IEM_MC_CALL_CIMPL_2(iemCImpl_sgdt, iEffSeg, GCPtrEffSrc);
297 IEM_MC_END();
298 return VINF_SUCCESS;
299}
300
301
302/** Opcode 0x0f 0x01 /0. */
303FNIEMOP_DEF(iemOp_Grp7_vmcall)
304{
305 IEMOP_BITCH_ABOUT_STUB();
306 return IEMOP_RAISE_INVALID_OPCODE();
307}
308
309
310/** Opcode 0x0f 0x01 /0. */
311FNIEMOP_DEF(iemOp_Grp7_vmlaunch)
312{
313 IEMOP_BITCH_ABOUT_STUB();
314 return IEMOP_RAISE_INVALID_OPCODE();
315}
316
317
318/** Opcode 0x0f 0x01 /0. */
319FNIEMOP_DEF(iemOp_Grp7_vmresume)
320{
321 IEMOP_BITCH_ABOUT_STUB();
322 return IEMOP_RAISE_INVALID_OPCODE();
323}
324
325
326/** Opcode 0x0f 0x01 /0. */
327FNIEMOP_DEF(iemOp_Grp7_vmxoff)
328{
329 IEMOP_BITCH_ABOUT_STUB();
330 return IEMOP_RAISE_INVALID_OPCODE();
331}
332
333
334/** Opcode 0x0f 0x01 /1. */
335FNIEMOP_DEF_1(iemOp_Grp7_sidt, uint8_t, bRm)
336{
337 IEMOP_MNEMONIC(sidt, "sidt Ms");
338 IEMOP_HLP_MIN_286();
339 IEMOP_HLP_64BIT_OP_SIZE();
340 IEM_MC_BEGIN(2, 1);
341 IEM_MC_ARG(uint8_t, iEffSeg, 0);
342 IEM_MC_ARG(RTGCPTR, GCPtrEffSrc, 1);
343 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
344 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
345 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
346 IEM_MC_CALL_CIMPL_2(iemCImpl_sidt, iEffSeg, GCPtrEffSrc);
347 IEM_MC_END();
348 return VINF_SUCCESS;
349}
350
351
352/** Opcode 0x0f 0x01 /1. */
353FNIEMOP_DEF(iemOp_Grp7_monitor)
354{
355 IEMOP_MNEMONIC(monitor, "monitor");
356 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX(); /** @todo Verify that monitor is allergic to lock prefixes. */
357 return IEM_MC_DEFER_TO_CIMPL_1(iemCImpl_monitor, pVCpu->iem.s.iEffSeg);
358}
359
360
361/** Opcode 0x0f 0x01 /1. */
362FNIEMOP_DEF(iemOp_Grp7_mwait)
363{
364 IEMOP_MNEMONIC(mwait, "mwait"); /** @todo Verify that mwait is allergic to lock prefixes. */
365 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
366 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_mwait);
367}
368
369
370/** Opcode 0x0f 0x01 /2. */
371FNIEMOP_DEF_1(iemOp_Grp7_lgdt, uint8_t, bRm)
372{
373 IEMOP_MNEMONIC(lgdt, "lgdt");
374 IEMOP_HLP_64BIT_OP_SIZE();
375 IEM_MC_BEGIN(3, 1);
376 IEM_MC_ARG(uint8_t, iEffSeg, 0);
377 IEM_MC_ARG(RTGCPTR, GCPtrEffSrc, 1);
378 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSizeArg,/*=*/pVCpu->iem.s.enmEffOpSize, 2);
379 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
380 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
381 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
382 IEM_MC_CALL_CIMPL_3(iemCImpl_lgdt, iEffSeg, GCPtrEffSrc, enmEffOpSizeArg);
383 IEM_MC_END();
384 return VINF_SUCCESS;
385}
386
387
388/** Opcode 0x0f 0x01 0xd0. */
389FNIEMOP_DEF(iemOp_Grp7_xgetbv)
390{
391 IEMOP_MNEMONIC(xgetbv, "xgetbv");
392 if (IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fXSaveRstor)
393 {
394 IEMOP_HLP_DONE_DECODING_NO_LOCK_REPZ_OR_REPNZ_PREFIXES();
395 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_xgetbv);
396 }
397 return IEMOP_RAISE_INVALID_OPCODE();
398}
399
400
401/** Opcode 0x0f 0x01 0xd1. */
402FNIEMOP_DEF(iemOp_Grp7_xsetbv)
403{
404 IEMOP_MNEMONIC(xsetbv, "xsetbv");
405 if (IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fXSaveRstor)
406 {
407 IEMOP_HLP_DONE_DECODING_NO_LOCK_REPZ_OR_REPNZ_PREFIXES();
408 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_xsetbv);
409 }
410 return IEMOP_RAISE_INVALID_OPCODE();
411}
412
413
414/** Opcode 0x0f 0x01 /3. */
415FNIEMOP_DEF_1(iemOp_Grp7_lidt, uint8_t, bRm)
416{
417 IEMOP_MNEMONIC(lidt, "lidt");
418 IEMMODE enmEffOpSize = pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT
419 ? IEMMODE_64BIT
420 : pVCpu->iem.s.enmEffOpSize;
421 IEM_MC_BEGIN(3, 1);
422 IEM_MC_ARG(uint8_t, iEffSeg, 0);
423 IEM_MC_ARG(RTGCPTR, GCPtrEffSrc, 1);
424 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSizeArg,/*=*/enmEffOpSize, 2);
425 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
426 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
427 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
428 IEM_MC_CALL_CIMPL_3(iemCImpl_lidt, iEffSeg, GCPtrEffSrc, enmEffOpSizeArg);
429 IEM_MC_END();
430 return VINF_SUCCESS;
431}
432
433
434#ifdef VBOX_WITH_NESTED_HWVIRT
435/** Opcode 0x0f 0x01 0xd8. */
436FNIEMOP_DEF(iemOp_Grp7_Amd_vmrun)
437{
438 IEMOP_MNEMONIC(vmrun, "vmrun");
439 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_vmrun);
440}
441
442/** Opcode 0x0f 0x01 0xd9. */
443FNIEMOP_DEF(iemOp_Grp7_Amd_vmmcall)
444{
445 IEMOP_MNEMONIC(vmmcall, "vmmcall");
446 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_vmmcall);
447}
448
449
450/** Opcode 0x0f 0x01 0xda. */
451FNIEMOP_DEF(iemOp_Grp7_Amd_vmload)
452{
453 IEMOP_MNEMONIC(vmload, "vmload");
454 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_vmload);
455}
456
457
458/** Opcode 0x0f 0x01 0xdb. */
459FNIEMOP_DEF(iemOp_Grp7_Amd_vmsave)
460{
461 IEMOP_MNEMONIC(vmsave, "vmsave");
462 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_vmsave);
463}
464
465
466/** Opcode 0x0f 0x01 0xdc. */
467FNIEMOP_DEF(iemOp_Grp7_Amd_stgi)
468{
469 IEMOP_MNEMONIC(stgi, "stgi");
470 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_stgi);
471}
472
473
474/** Opcode 0x0f 0x01 0xdd. */
475FNIEMOP_DEF(iemOp_Grp7_Amd_clgi)
476{
477 IEMOP_MNEMONIC(clgi, "clgi");
478 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_clgi);
479}
480
481
482/** Opcode 0x0f 0x01 0xdf. */
483FNIEMOP_DEF(iemOp_Grp7_Amd_invlpga)
484{
485 IEMOP_MNEMONIC(invlpga, "invlpga");
486 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_invlpga);
487}
488
489
490/** Opcode 0x0f 0x01 0xde. */
491FNIEMOP_DEF(iemOp_Grp7_Amd_skinit)
492{
493 IEMOP_MNEMONIC(skinit, "skinit");
494 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_skinit);
495}
496#else
497/** Opcode 0x0f 0x01 0xd8. */
498FNIEMOP_UD_STUB(iemOp_Grp7_Amd_vmrun);
499
500/** Opcode 0x0f 0x01 0xd9. */
501FNIEMOP_UD_STUB(iemOp_Grp7_Amd_vmmcall);
502
503/** Opcode 0x0f 0x01 0xda. */
504FNIEMOP_UD_STUB(iemOp_Grp7_Amd_vmload);
505
506/** Opcode 0x0f 0x01 0xdb. */
507FNIEMOP_UD_STUB(iemOp_Grp7_Amd_vmsave);
508
509/** Opcode 0x0f 0x01 0xdc. */
510FNIEMOP_UD_STUB(iemOp_Grp7_Amd_stgi);
511
512/** Opcode 0x0f 0x01 0xdd. */
513FNIEMOP_UD_STUB(iemOp_Grp7_Amd_clgi);
514
515/** Opcode 0x0f 0x01 0xdf. */
516FNIEMOP_UD_STUB(iemOp_Grp7_Amd_invlpga);
517
518/** Opcode 0x0f 0x01 0xde. */
519FNIEMOP_UD_STUB(iemOp_Grp7_Amd_skinit);
520#endif /* VBOX_WITH_NESTED_HWVIRT */
521
522/** Opcode 0x0f 0x01 /4. */
523FNIEMOP_DEF_1(iemOp_Grp7_smsw, uint8_t, bRm)
524{
525 IEMOP_MNEMONIC(smsw, "smsw");
526 IEMOP_HLP_MIN_286();
527 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
528 {
529 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
530 IEMOP_HLP_SVM_READ_CR_INTERCEPT(pVCpu, /*cr*/ 0, 0 /* uExitInfo1 */, 0 /* uExitInfo2 */);
531 switch (pVCpu->iem.s.enmEffOpSize)
532 {
533 case IEMMODE_16BIT:
534 IEM_MC_BEGIN(0, 1);
535 IEM_MC_LOCAL(uint16_t, u16Tmp);
536 IEM_MC_FETCH_CR0_U16(u16Tmp);
537 if (IEM_GET_TARGET_CPU(pVCpu) > IEMTARGETCPU_386)
538 { /* likely */ }
539 else if (IEM_GET_TARGET_CPU(pVCpu) >= IEMTARGETCPU_386)
540 IEM_MC_OR_LOCAL_U16(u16Tmp, 0xffe0);
541 else
542 IEM_MC_OR_LOCAL_U16(u16Tmp, 0xfff0);
543 IEM_MC_STORE_GREG_U16((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u16Tmp);
544 IEM_MC_ADVANCE_RIP();
545 IEM_MC_END();
546 return VINF_SUCCESS;
547
548 case IEMMODE_32BIT:
549 IEM_MC_BEGIN(0, 1);
550 IEM_MC_LOCAL(uint32_t, u32Tmp);
551 IEM_MC_FETCH_CR0_U32(u32Tmp);
552 IEM_MC_STORE_GREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u32Tmp);
553 IEM_MC_ADVANCE_RIP();
554 IEM_MC_END();
555 return VINF_SUCCESS;
556
557 case IEMMODE_64BIT:
558 IEM_MC_BEGIN(0, 1);
559 IEM_MC_LOCAL(uint64_t, u64Tmp);
560 IEM_MC_FETCH_CR0_U64(u64Tmp);
561 IEM_MC_STORE_GREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u64Tmp);
562 IEM_MC_ADVANCE_RIP();
563 IEM_MC_END();
564 return VINF_SUCCESS;
565
566 IEM_NOT_REACHED_DEFAULT_CASE_RET();
567 }
568 }
569 else
570 {
571 /* Ignore operand size here, memory refs are always 16-bit. */
572 IEM_MC_BEGIN(0, 2);
573 IEM_MC_LOCAL(uint16_t, u16Tmp);
574 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
575 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
576 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
577 IEMOP_HLP_SVM_READ_CR_INTERCEPT(pVCpu, /*cr*/ 0, 0 /* uExitInfo1 */, 0 /* uExitInfo2 */);
578 IEM_MC_FETCH_CR0_U16(u16Tmp);
579 if (IEM_GET_TARGET_CPU(pVCpu) > IEMTARGETCPU_386)
580 { /* likely */ }
581 else if (pVCpu->iem.s.uTargetCpu >= IEMTARGETCPU_386)
582 IEM_MC_OR_LOCAL_U16(u16Tmp, 0xffe0);
583 else
584 IEM_MC_OR_LOCAL_U16(u16Tmp, 0xfff0);
585 IEM_MC_STORE_MEM_U16(pVCpu->iem.s.iEffSeg, GCPtrEffDst, u16Tmp);
586 IEM_MC_ADVANCE_RIP();
587 IEM_MC_END();
588 return VINF_SUCCESS;
589 }
590}
591
592
593/** Opcode 0x0f 0x01 /6. */
594FNIEMOP_DEF_1(iemOp_Grp7_lmsw, uint8_t, bRm)
595{
596 /* The operand size is effectively ignored, all is 16-bit and only the
597 lower 3-bits are used. */
598 IEMOP_MNEMONIC(lmsw, "lmsw");
599 IEMOP_HLP_MIN_286();
600 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
601 {
602 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
603 IEM_MC_BEGIN(1, 0);
604 IEM_MC_ARG(uint16_t, u16Tmp, 0);
605 IEM_MC_FETCH_GREG_U16(u16Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
606 IEM_MC_CALL_CIMPL_1(iemCImpl_lmsw, u16Tmp);
607 IEM_MC_END();
608 }
609 else
610 {
611 IEM_MC_BEGIN(1, 1);
612 IEM_MC_ARG(uint16_t, u16Tmp, 0);
613 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
614 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
615 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
616 IEM_MC_FETCH_MEM_U16(u16Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
617 IEM_MC_CALL_CIMPL_1(iemCImpl_lmsw, u16Tmp);
618 IEM_MC_END();
619 }
620 return VINF_SUCCESS;
621}
622
623
624/** Opcode 0x0f 0x01 /7. */
625FNIEMOP_DEF_1(iemOp_Grp7_invlpg, uint8_t, bRm)
626{
627 IEMOP_MNEMONIC(invlpg, "invlpg");
628 IEMOP_HLP_MIN_486();
629 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
630 IEM_MC_BEGIN(1, 1);
631 IEM_MC_ARG(RTGCPTR, GCPtrEffDst, 0);
632 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
633 IEM_MC_CALL_CIMPL_1(iemCImpl_invlpg, GCPtrEffDst);
634 IEM_MC_END();
635 return VINF_SUCCESS;
636}
637
638
639/** Opcode 0x0f 0x01 /7. */
640FNIEMOP_DEF(iemOp_Grp7_swapgs)
641{
642 IEMOP_MNEMONIC(swapgs, "swapgs");
643 IEMOP_HLP_ONLY_64BIT();
644 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
645 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_swapgs);
646}
647
648
649/** Opcode 0x0f 0x01 /7. */
650FNIEMOP_DEF(iemOp_Grp7_rdtscp)
651{
652 IEMOP_MNEMONIC(rdtscp, "rdtscp");
653 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
654 /** @todo SVM intercept removal from here. */
655 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_RDTSCP, SVM_EXIT_RDTSCP, 0, 0);
656 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_rdtscp);
657}
658
659
660/**
661 * Group 7 jump table, memory variant.
662 */
663IEM_STATIC const PFNIEMOPRM g_apfnGroup7Mem[8] =
664{
665 iemOp_Grp7_sgdt,
666 iemOp_Grp7_sidt,
667 iemOp_Grp7_lgdt,
668 iemOp_Grp7_lidt,
669 iemOp_Grp7_smsw,
670 iemOp_InvalidWithRM,
671 iemOp_Grp7_lmsw,
672 iemOp_Grp7_invlpg
673};
674
675
676/** Opcode 0x0f 0x01. */
677FNIEMOP_DEF(iemOp_Grp7)
678{
679 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
680 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
681 return FNIEMOP_CALL_1(g_apfnGroup7Mem[(bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK], bRm);
682
683 switch ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK)
684 {
685 case 0:
686 switch (bRm & X86_MODRM_RM_MASK)
687 {
688 case 1: return FNIEMOP_CALL(iemOp_Grp7_vmcall);
689 case 2: return FNIEMOP_CALL(iemOp_Grp7_vmlaunch);
690 case 3: return FNIEMOP_CALL(iemOp_Grp7_vmresume);
691 case 4: return FNIEMOP_CALL(iemOp_Grp7_vmxoff);
692 }
693 return IEMOP_RAISE_INVALID_OPCODE();
694
695 case 1:
696 switch (bRm & X86_MODRM_RM_MASK)
697 {
698 case 0: return FNIEMOP_CALL(iemOp_Grp7_monitor);
699 case 1: return FNIEMOP_CALL(iemOp_Grp7_mwait);
700 }
701 return IEMOP_RAISE_INVALID_OPCODE();
702
703 case 2:
704 switch (bRm & X86_MODRM_RM_MASK)
705 {
706 case 0: return FNIEMOP_CALL(iemOp_Grp7_xgetbv);
707 case 1: return FNIEMOP_CALL(iemOp_Grp7_xsetbv);
708 }
709 return IEMOP_RAISE_INVALID_OPCODE();
710
711 case 3:
712 switch (bRm & X86_MODRM_RM_MASK)
713 {
714 case 0: return FNIEMOP_CALL(iemOp_Grp7_Amd_vmrun);
715 case 1: return FNIEMOP_CALL(iemOp_Grp7_Amd_vmmcall);
716 case 2: return FNIEMOP_CALL(iemOp_Grp7_Amd_vmload);
717 case 3: return FNIEMOP_CALL(iemOp_Grp7_Amd_vmsave);
718 case 4: return FNIEMOP_CALL(iemOp_Grp7_Amd_stgi);
719 case 5: return FNIEMOP_CALL(iemOp_Grp7_Amd_clgi);
720 case 6: return FNIEMOP_CALL(iemOp_Grp7_Amd_skinit);
721 case 7: return FNIEMOP_CALL(iemOp_Grp7_Amd_invlpga);
722 IEM_NOT_REACHED_DEFAULT_CASE_RET();
723 }
724
725 case 4:
726 return FNIEMOP_CALL_1(iemOp_Grp7_smsw, bRm);
727
728 case 5:
729 return IEMOP_RAISE_INVALID_OPCODE();
730
731 case 6:
732 return FNIEMOP_CALL_1(iemOp_Grp7_lmsw, bRm);
733
734 case 7:
735 switch (bRm & X86_MODRM_RM_MASK)
736 {
737 case 0: return FNIEMOP_CALL(iemOp_Grp7_swapgs);
738 case 1: return FNIEMOP_CALL(iemOp_Grp7_rdtscp);
739 }
740 return IEMOP_RAISE_INVALID_OPCODE();
741
742 IEM_NOT_REACHED_DEFAULT_CASE_RET();
743 }
744}
745
746/** Opcode 0x0f 0x00 /3. */
747FNIEMOP_DEF_1(iemOpCommonLarLsl_Gv_Ew, bool, fIsLar)
748{
749 IEMOP_HLP_NO_REAL_OR_V86_MODE();
750 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
751
752 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
753 {
754 IEMOP_HLP_DECODED_NL_2(fIsLar ? OP_LAR : OP_LSL, IEMOPFORM_RM_REG, OP_PARM_Gv, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
755 switch (pVCpu->iem.s.enmEffOpSize)
756 {
757 case IEMMODE_16BIT:
758 {
759 IEM_MC_BEGIN(3, 0);
760 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
761 IEM_MC_ARG(uint16_t, u16Sel, 1);
762 IEM_MC_ARG_CONST(bool, fIsLarArg, fIsLar, 2);
763
764 IEM_MC_REF_GREG_U16(pu16Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
765 IEM_MC_FETCH_GREG_U16(u16Sel, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
766 IEM_MC_CALL_CIMPL_3(iemCImpl_LarLsl_u16, pu16Dst, u16Sel, fIsLarArg);
767
768 IEM_MC_END();
769 return VINF_SUCCESS;
770 }
771
772 case IEMMODE_32BIT:
773 case IEMMODE_64BIT:
774 {
775 IEM_MC_BEGIN(3, 0);
776 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
777 IEM_MC_ARG(uint16_t, u16Sel, 1);
778 IEM_MC_ARG_CONST(bool, fIsLarArg, fIsLar, 2);
779
780 IEM_MC_REF_GREG_U64(pu64Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
781 IEM_MC_FETCH_GREG_U16(u16Sel, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
782 IEM_MC_CALL_CIMPL_3(iemCImpl_LarLsl_u64, pu64Dst, u16Sel, fIsLarArg);
783
784 IEM_MC_END();
785 return VINF_SUCCESS;
786 }
787
788 IEM_NOT_REACHED_DEFAULT_CASE_RET();
789 }
790 }
791 else
792 {
793 switch (pVCpu->iem.s.enmEffOpSize)
794 {
795 case IEMMODE_16BIT:
796 {
797 IEM_MC_BEGIN(3, 1);
798 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
799 IEM_MC_ARG(uint16_t, u16Sel, 1);
800 IEM_MC_ARG_CONST(bool, fIsLarArg, fIsLar, 2);
801 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
802
803 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
804 IEMOP_HLP_DECODED_NL_2(fIsLar ? OP_LAR : OP_LSL, IEMOPFORM_RM_MEM, OP_PARM_Gv, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
805
806 IEM_MC_FETCH_MEM_U16(u16Sel, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
807 IEM_MC_REF_GREG_U16(pu16Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
808 IEM_MC_CALL_CIMPL_3(iemCImpl_LarLsl_u16, pu16Dst, u16Sel, fIsLarArg);
809
810 IEM_MC_END();
811 return VINF_SUCCESS;
812 }
813
814 case IEMMODE_32BIT:
815 case IEMMODE_64BIT:
816 {
817 IEM_MC_BEGIN(3, 1);
818 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
819 IEM_MC_ARG(uint16_t, u16Sel, 1);
820 IEM_MC_ARG_CONST(bool, fIsLarArg, fIsLar, 2);
821 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
822
823 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
824 IEMOP_HLP_DECODED_NL_2(fIsLar ? OP_LAR : OP_LSL, IEMOPFORM_RM_MEM, OP_PARM_Gv, OP_PARM_Ew, DISOPTYPE_DANGEROUS | DISOPTYPE_PRIVILEGED_NOTRAP);
825/** @todo testcase: make sure it's a 16-bit read. */
826
827 IEM_MC_FETCH_MEM_U16(u16Sel, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
828 IEM_MC_REF_GREG_U64(pu64Dst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
829 IEM_MC_CALL_CIMPL_3(iemCImpl_LarLsl_u64, pu64Dst, u16Sel, fIsLarArg);
830
831 IEM_MC_END();
832 return VINF_SUCCESS;
833 }
834
835 IEM_NOT_REACHED_DEFAULT_CASE_RET();
836 }
837 }
838}
839
840
841
842/** Opcode 0x0f 0x02. */
843FNIEMOP_DEF(iemOp_lar_Gv_Ew)
844{
845 IEMOP_MNEMONIC(lar, "lar Gv,Ew");
846 return FNIEMOP_CALL_1(iemOpCommonLarLsl_Gv_Ew, true);
847}
848
849
850/** Opcode 0x0f 0x03. */
851FNIEMOP_DEF(iemOp_lsl_Gv_Ew)
852{
853 IEMOP_MNEMONIC(lsl, "lsl Gv,Ew");
854 return FNIEMOP_CALL_1(iemOpCommonLarLsl_Gv_Ew, false);
855}
856
857
858/** Opcode 0x0f 0x05. */
859FNIEMOP_DEF(iemOp_syscall)
860{
861 IEMOP_MNEMONIC(syscall, "syscall"); /** @todo 286 LOADALL */
862 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
863 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_syscall);
864}
865
866
867/** Opcode 0x0f 0x06. */
868FNIEMOP_DEF(iemOp_clts)
869{
870 IEMOP_MNEMONIC(clts, "clts");
871 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
872 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_clts);
873}
874
875
876/** Opcode 0x0f 0x07. */
877FNIEMOP_DEF(iemOp_sysret)
878{
879 IEMOP_MNEMONIC(sysret, "sysret"); /** @todo 386 LOADALL */
880 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
881 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_sysret);
882}
883
884
885/** Opcode 0x0f 0x08. */
886FNIEMOP_DEF(iemOp_invd)
887{
888 IEMOP_MNEMONIC(invd, "invd");
889#ifdef VBOX_WITH_NESTED_HWVIRT
890 IEM_MC_RAISE_GP0_IF_CPL_NOT_ZERO();
891 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_INVD, SVM_EXIT_INVD, 0, 0);
892#else
893 RT_NOREF_PV(pVCpu);
894#endif
895 /** @todo implement invd for the regular case (above only handles nested SVM
896 * exits). */
897 IEMOP_BITCH_ABOUT_STUB();
898 return VERR_IEM_INSTR_NOT_IMPLEMENTED;
899}
900
901// IEMOP_HLP_MIN_486();
902
903
904/** Opcode 0x0f 0x09. */
905FNIEMOP_DEF(iemOp_wbinvd)
906{
907 IEMOP_MNEMONIC(wbinvd, "wbinvd");
908 IEMOP_HLP_MIN_486();
909 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
910 IEM_MC_BEGIN(0, 0);
911 IEM_MC_RAISE_GP0_IF_CPL_NOT_ZERO();
912 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_WBINVD, SVM_EXIT_WBINVD, 0, 0);
913 IEM_MC_ADVANCE_RIP();
914 IEM_MC_END();
915 return VINF_SUCCESS; /* ignore for now */
916}
917
918
919/** Opcode 0x0f 0x0b. */
920FNIEMOP_DEF(iemOp_ud2)
921{
922 IEMOP_MNEMONIC(ud2, "ud2");
923 return IEMOP_RAISE_INVALID_OPCODE();
924}
925
926/** Opcode 0x0f 0x0d. */
927FNIEMOP_DEF(iemOp_nop_Ev_GrpP)
928{
929 /* AMD prefetch group, Intel implements this as NOP Ev (and so do we). */
930 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->f3DNowPrefetch)
931 {
932 IEMOP_MNEMONIC(GrpPNotSupported, "GrpP");
933 return IEMOP_RAISE_INVALID_OPCODE();
934 }
935
936 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
937 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
938 {
939 IEMOP_MNEMONIC(GrpPInvalid, "GrpP");
940 return IEMOP_RAISE_INVALID_OPCODE();
941 }
942
943 switch ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK)
944 {
945 case 2: /* Aliased to /0 for the time being. */
946 case 4: /* Aliased to /0 for the time being. */
947 case 5: /* Aliased to /0 for the time being. */
948 case 6: /* Aliased to /0 for the time being. */
949 case 7: /* Aliased to /0 for the time being. */
950 case 0: IEMOP_MNEMONIC(prefetch, "prefetch"); break;
951 case 1: IEMOP_MNEMONIC(prefetchw_1, "prefetchw"); break;
952 case 3: IEMOP_MNEMONIC(prefetchw_3, "prefetchw"); break;
953 IEM_NOT_REACHED_DEFAULT_CASE_RET();
954 }
955
956 IEM_MC_BEGIN(0, 1);
957 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
958 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
959 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
960 /* Currently a NOP. */
961 NOREF(GCPtrEffSrc);
962 IEM_MC_ADVANCE_RIP();
963 IEM_MC_END();
964 return VINF_SUCCESS;
965}
966
967
968/** Opcode 0x0f 0x0e. */
969FNIEMOP_STUB(iemOp_femms);
970
971
972/** Opcode 0x0f 0x0f. */
973FNIEMOP_DEF(iemOp_3Dnow)
974{
975 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->f3DNow)
976 {
977 IEMOP_MNEMONIC(Inv3Dnow, "3Dnow");
978 return IEMOP_RAISE_INVALID_OPCODE();
979 }
980
981#ifdef IEM_WITH_3DNOW
982 /* This is pretty sparse, use switch instead of table. */
983 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
984 return FNIEMOP_CALL_1(iemOp_3DNowDispatcher, b);
985#else
986 IEMOP_BITCH_ABOUT_STUB();
987 return VERR_IEM_INSTR_NOT_IMPLEMENTED;
988#endif
989}
990
991
992/**
993 * @opcode 0x10
994 * @oppfx none
995 * @opcpuid sse
996 * @opgroup og_sse_simdfp_datamove
997 * @opxcpttype 4UA
998 * @optest op1=1 op2=2 -> op1=2
999 * @optest op1=0 op2=-22 -> op1=-22
1000 */
1001FNIEMOP_DEF(iemOp_movups_Vps_Wps)
1002{
1003 IEMOP_MNEMONIC2(RM, MOVUPS, movups, Vps_WO, Wps, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1004 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1005 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1006 {
1007 /*
1008 * Register, register.
1009 */
1010 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1011 IEM_MC_BEGIN(0, 0);
1012 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1013 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1014 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
1015 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1016 IEM_MC_ADVANCE_RIP();
1017 IEM_MC_END();
1018 }
1019 else
1020 {
1021 /*
1022 * Memory, register.
1023 */
1024 IEM_MC_BEGIN(0, 2);
1025 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
1026 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1027
1028 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1029 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1030 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1031 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1032
1033 IEM_MC_FETCH_MEM_U128(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1034 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1035
1036 IEM_MC_ADVANCE_RIP();
1037 IEM_MC_END();
1038 }
1039 return VINF_SUCCESS;
1040
1041}
1042
1043
1044/**
1045 * @opcode 0x10
1046 * @oppfx 0x66
1047 * @opcpuid sse2
1048 * @opgroup og_sse2_pcksclr_datamove
1049 * @opxcpttype 4UA
1050 * @optest op1=1 op2=2 -> op1=2
1051 * @optest op1=0 op2=-42 -> op1=-42
1052 */
1053FNIEMOP_DEF(iemOp_movupd_Vpd_Wpd)
1054{
1055 IEMOP_MNEMONIC2(RM, MOVUPD, movupd, Vpd_WO, Wpd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1056 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1057 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1058 {
1059 /*
1060 * Register, register.
1061 */
1062 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1063 IEM_MC_BEGIN(0, 0);
1064 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1065 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1066 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
1067 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1068 IEM_MC_ADVANCE_RIP();
1069 IEM_MC_END();
1070 }
1071 else
1072 {
1073 /*
1074 * Memory, register.
1075 */
1076 IEM_MC_BEGIN(0, 2);
1077 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
1078 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1079
1080 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1081 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1082 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1083 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1084
1085 IEM_MC_FETCH_MEM_U128(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1086 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1087
1088 IEM_MC_ADVANCE_RIP();
1089 IEM_MC_END();
1090 }
1091 return VINF_SUCCESS;
1092}
1093
1094
1095/**
1096 * @opcode 0x10
1097 * @oppfx 0xf3
1098 * @opcpuid sse
1099 * @opgroup og_sse_simdfp_datamove
1100 * @opxcpttype 5
1101 * @optest op1=1 op2=2 -> op1=2
1102 * @optest op1=0 op2=-22 -> op1=-22
1103 */
1104FNIEMOP_DEF(iemOp_movss_Vss_Wss)
1105{
1106 IEMOP_MNEMONIC2(RM, MOVSS, movss, VssZx_WO, Wss, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1107 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1108 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1109 {
1110 /*
1111 * Register, register.
1112 */
1113 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1114 IEM_MC_BEGIN(0, 1);
1115 IEM_MC_LOCAL(uint32_t, uSrc);
1116
1117 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1118 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1119 IEM_MC_FETCH_XREG_U32(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1120 IEM_MC_STORE_XREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1121
1122 IEM_MC_ADVANCE_RIP();
1123 IEM_MC_END();
1124 }
1125 else
1126 {
1127 /*
1128 * Memory, register.
1129 */
1130 IEM_MC_BEGIN(0, 2);
1131 IEM_MC_LOCAL(uint32_t, uSrc);
1132 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1133
1134 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1135 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1136 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1137 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1138
1139 IEM_MC_FETCH_MEM_U32(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1140 IEM_MC_STORE_XREG_U32_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1141
1142 IEM_MC_ADVANCE_RIP();
1143 IEM_MC_END();
1144 }
1145 return VINF_SUCCESS;
1146}
1147
1148
1149/**
1150 * @opcode 0x10
1151 * @oppfx 0xf2
1152 * @opcpuid sse2
1153 * @opgroup og_sse2_pcksclr_datamove
1154 * @opxcpttype 5
1155 * @optest op1=1 op2=2 -> op1=2
1156 * @optest op1=0 op2=-42 -> op1=-42
1157 */
1158FNIEMOP_DEF(iemOp_movsd_Vsd_Wsd)
1159{
1160 IEMOP_MNEMONIC2(RM, MOVSD, movsd, VsdZx_WO, Wsd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1161 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1162 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1163 {
1164 /*
1165 * Register, register.
1166 */
1167 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1168 IEM_MC_BEGIN(0, 1);
1169 IEM_MC_LOCAL(uint64_t, uSrc);
1170
1171 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1172 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1173 IEM_MC_FETCH_XREG_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1174 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1175
1176 IEM_MC_ADVANCE_RIP();
1177 IEM_MC_END();
1178 }
1179 else
1180 {
1181 /*
1182 * Memory, register.
1183 */
1184 IEM_MC_BEGIN(0, 2);
1185 IEM_MC_LOCAL(uint64_t, uSrc);
1186 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1187
1188 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1189 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1190 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1191 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1192
1193 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1194 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1195
1196 IEM_MC_ADVANCE_RIP();
1197 IEM_MC_END();
1198 }
1199 return VINF_SUCCESS;
1200}
1201
1202
1203/**
1204 * @opcode 0x11
1205 * @oppfx none
1206 * @opcpuid sse
1207 * @opgroup og_sse_simdfp_datamove
1208 * @opxcpttype 4UA
1209 * @optest op1=1 op2=2 -> op1=2
1210 * @optest op1=0 op2=-42 -> op1=-42
1211 */
1212FNIEMOP_DEF(iemOp_movups_Wps_Vps)
1213{
1214 IEMOP_MNEMONIC2(MR, MOVUPS, movups, Wps_WO, Vps, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1215 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1216 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1217 {
1218 /*
1219 * Register, register.
1220 */
1221 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1222 IEM_MC_BEGIN(0, 0);
1223 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1224 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1225 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
1226 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1227 IEM_MC_ADVANCE_RIP();
1228 IEM_MC_END();
1229 }
1230 else
1231 {
1232 /*
1233 * Memory, register.
1234 */
1235 IEM_MC_BEGIN(0, 2);
1236 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
1237 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1238
1239 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1240 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1241 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1242 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1243
1244 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1245 IEM_MC_STORE_MEM_U128(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1246
1247 IEM_MC_ADVANCE_RIP();
1248 IEM_MC_END();
1249 }
1250 return VINF_SUCCESS;
1251}
1252
1253
1254/**
1255 * @opcode 0x11
1256 * @oppfx 0x66
1257 * @opcpuid sse2
1258 * @opgroup og_sse2_pcksclr_datamove
1259 * @opxcpttype 4UA
1260 * @optest op1=1 op2=2 -> op1=2
1261 * @optest op1=0 op2=-42 -> op1=-42
1262 */
1263FNIEMOP_DEF(iemOp_movupd_Wpd_Vpd)
1264{
1265 IEMOP_MNEMONIC2(MR, MOVUPD, movupd, Wpd_WO, Vpd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1266 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1267 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1268 {
1269 /*
1270 * Register, register.
1271 */
1272 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1273 IEM_MC_BEGIN(0, 0);
1274 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1275 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1276 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
1277 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1278 IEM_MC_ADVANCE_RIP();
1279 IEM_MC_END();
1280 }
1281 else
1282 {
1283 /*
1284 * Memory, register.
1285 */
1286 IEM_MC_BEGIN(0, 2);
1287 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
1288 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1289
1290 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1291 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1292 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1293 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1294
1295 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1296 IEM_MC_STORE_MEM_U128(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1297
1298 IEM_MC_ADVANCE_RIP();
1299 IEM_MC_END();
1300 }
1301 return VINF_SUCCESS;
1302}
1303
1304
1305/**
1306 * @opcode 0x11
1307 * @oppfx 0xf3
1308 * @opcpuid sse
1309 * @opgroup og_sse_simdfp_datamove
1310 * @opxcpttype 5
1311 * @optest op1=1 op2=2 -> op1=2
1312 * @optest op1=0 op2=-22 -> op1=-22
1313 */
1314FNIEMOP_DEF(iemOp_movss_Wss_Vss)
1315{
1316 IEMOP_MNEMONIC2(MR, MOVSS, movss, Wss_WO, Vss, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1317 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1318 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1319 {
1320 /*
1321 * Register, register.
1322 */
1323 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1324 IEM_MC_BEGIN(0, 1);
1325 IEM_MC_LOCAL(uint32_t, uSrc);
1326
1327 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1328 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1329 IEM_MC_FETCH_XREG_U32(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1330 IEM_MC_STORE_XREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, uSrc);
1331
1332 IEM_MC_ADVANCE_RIP();
1333 IEM_MC_END();
1334 }
1335 else
1336 {
1337 /*
1338 * Memory, register.
1339 */
1340 IEM_MC_BEGIN(0, 2);
1341 IEM_MC_LOCAL(uint32_t, uSrc);
1342 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1343
1344 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1345 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1346 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1347 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1348
1349 IEM_MC_FETCH_XREG_U32(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1350 IEM_MC_STORE_MEM_U32(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1351
1352 IEM_MC_ADVANCE_RIP();
1353 IEM_MC_END();
1354 }
1355 return VINF_SUCCESS;
1356}
1357
1358
1359/**
1360 * @opcode 0x11
1361 * @oppfx 0xf2
1362 * @opcpuid sse2
1363 * @opgroup og_sse2_pcksclr_datamove
1364 * @opxcpttype 5
1365 * @optest op1=1 op2=2 -> op1=2
1366 * @optest op1=0 op2=-42 -> op1=-42
1367 */
1368FNIEMOP_DEF(iemOp_movsd_Wsd_Vsd)
1369{
1370 IEMOP_MNEMONIC2(MR, MOVSD, movsd, Wsd_WO, Vsd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1371 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1372 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1373 {
1374 /*
1375 * Register, register.
1376 */
1377 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1378 IEM_MC_BEGIN(0, 1);
1379 IEM_MC_LOCAL(uint64_t, uSrc);
1380
1381 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1382 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1383 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1384 IEM_MC_STORE_XREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, uSrc);
1385
1386 IEM_MC_ADVANCE_RIP();
1387 IEM_MC_END();
1388 }
1389 else
1390 {
1391 /*
1392 * Memory, register.
1393 */
1394 IEM_MC_BEGIN(0, 2);
1395 IEM_MC_LOCAL(uint64_t, uSrc);
1396 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1397
1398 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1399 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1400 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1401 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1402
1403 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1404 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1405
1406 IEM_MC_ADVANCE_RIP();
1407 IEM_MC_END();
1408 }
1409 return VINF_SUCCESS;
1410}
1411
1412
1413FNIEMOP_DEF(iemOp_movlps_Vq_Mq__movhlps)
1414{
1415 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1416 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1417 {
1418 /**
1419 * @opcode 0x12
1420 * @opcodesub 11 mr/reg
1421 * @oppfx none
1422 * @opcpuid sse
1423 * @opgroup og_sse_simdfp_datamove
1424 * @opxcpttype 5
1425 * @optest op1=1 op2=2 -> op1=2
1426 * @optest op1=0 op2=-42 -> op1=-42
1427 */
1428 IEMOP_MNEMONIC2(RM_REG, MOVHLPS, movhlps, Vq_WO, UqHi, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1429
1430 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1431 IEM_MC_BEGIN(0, 1);
1432 IEM_MC_LOCAL(uint64_t, uSrc);
1433
1434 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1435 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1436 IEM_MC_FETCH_XREG_HI_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1437 IEM_MC_STORE_XREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1438
1439 IEM_MC_ADVANCE_RIP();
1440 IEM_MC_END();
1441 }
1442 else
1443 {
1444 /**
1445 * @opdone
1446 * @opcode 0x12
1447 * @opcodesub !11 mr/reg
1448 * @oppfx none
1449 * @opcpuid sse
1450 * @opgroup og_sse_simdfp_datamove
1451 * @opxcpttype 5
1452 * @optest op1=1 op2=2 -> op1=2
1453 * @optest op1=0 op2=-42 -> op1=-42
1454 * @opfunction iemOp_movlps_Vq_Mq__vmovhlps
1455 */
1456 IEMOP_MNEMONIC2(RM_MEM, MOVLPS, movlps, Vq_WO, Mq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1457
1458 IEM_MC_BEGIN(0, 2);
1459 IEM_MC_LOCAL(uint64_t, uSrc);
1460 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1461
1462 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1463 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1464 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1465 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1466
1467 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1468 IEM_MC_STORE_XREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1469
1470 IEM_MC_ADVANCE_RIP();
1471 IEM_MC_END();
1472 }
1473 return VINF_SUCCESS;
1474}
1475
1476
1477/**
1478 * @opcode 0x12
1479 * @opcodesub !11 mr/reg
1480 * @oppfx 0x66
1481 * @opcpuid sse2
1482 * @opgroup og_sse2_pcksclr_datamove
1483 * @opxcpttype 5
1484 * @optest op1=1 op2=2 -> op1=2
1485 * @optest op1=0 op2=-42 -> op1=-42
1486 */
1487FNIEMOP_DEF(iemOp_movlpd_Vq_Mq)
1488{
1489 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1490 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
1491 {
1492 IEMOP_MNEMONIC2(RM_MEM, MOVLPD, movlpd, Vq_WO, Mq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1493
1494 IEM_MC_BEGIN(0, 2);
1495 IEM_MC_LOCAL(uint64_t, uSrc);
1496 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1497
1498 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1499 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1500 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1501 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1502
1503 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1504 IEM_MC_STORE_XREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1505
1506 IEM_MC_ADVANCE_RIP();
1507 IEM_MC_END();
1508 return VINF_SUCCESS;
1509 }
1510
1511 /**
1512 * @opdone
1513 * @opmnemonic ud660f12m3
1514 * @opcode 0x12
1515 * @opcodesub 11 mr/reg
1516 * @oppfx 0x66
1517 * @opunused immediate
1518 * @opcpuid sse
1519 * @optest ->
1520 */
1521 return IEMOP_RAISE_INVALID_OPCODE();
1522}
1523
1524
1525/**
1526 * @opcode 0x12
1527 * @oppfx 0xf3
1528 * @opcpuid sse3
1529 * @opgroup og_sse3_pcksclr_datamove
1530 * @opxcpttype 4
1531 * @optest op1=-1 op2=0xdddddddd00000002eeeeeeee00000001 ->
1532 * op1=0x00000002000000020000000100000001
1533 */
1534FNIEMOP_DEF(iemOp_movsldup_Vdq_Wdq)
1535{
1536 IEMOP_MNEMONIC2(RM, MOVSLDUP, movsldup, Vdq_WO, Wdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1537 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1538 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1539 {
1540 /*
1541 * Register, register.
1542 */
1543 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1544 IEM_MC_BEGIN(2, 0);
1545 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1546 IEM_MC_ARG(PCRTUINT128U, puSrc, 1);
1547
1548 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1549 IEM_MC_PREPARE_SSE_USAGE();
1550
1551 IEM_MC_REF_XREG_U128_CONST(puSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1552 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1553 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movsldup, puDst, puSrc);
1554
1555 IEM_MC_ADVANCE_RIP();
1556 IEM_MC_END();
1557 }
1558 else
1559 {
1560 /*
1561 * Register, memory.
1562 */
1563 IEM_MC_BEGIN(2, 2);
1564 IEM_MC_LOCAL(RTUINT128U, uSrc);
1565 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1566 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1567 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, puSrc, uSrc, 1);
1568
1569 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1570 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1571 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1572 IEM_MC_PREPARE_SSE_USAGE();
1573
1574 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1575 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1576 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movsldup, puDst, puSrc);
1577
1578 IEM_MC_ADVANCE_RIP();
1579 IEM_MC_END();
1580 }
1581 return VINF_SUCCESS;
1582}
1583
1584
1585/**
1586 * @opcode 0x12
1587 * @oppfx 0xf2
1588 * @opcpuid sse3
1589 * @opgroup og_sse3_pcksclr_datamove
1590 * @opxcpttype 5
1591 * @optest op1=-1 op2=0xddddddddeeeeeeee2222222211111111 ->
1592 * op1=0x22222222111111112222222211111111
1593 */
1594FNIEMOP_DEF(iemOp_movddup_Vdq_Wdq)
1595{
1596 IEMOP_MNEMONIC2(RM, MOVDDUP, movddup, Vdq_WO, Wdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1597 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1598 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1599 {
1600 /*
1601 * Register, register.
1602 */
1603 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1604 IEM_MC_BEGIN(2, 0);
1605 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1606 IEM_MC_ARG(uint64_t, uSrc, 1);
1607
1608 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1609 IEM_MC_PREPARE_SSE_USAGE();
1610
1611 IEM_MC_FETCH_XREG_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1612 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1613 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movddup, puDst, uSrc);
1614
1615 IEM_MC_ADVANCE_RIP();
1616 IEM_MC_END();
1617 }
1618 else
1619 {
1620 /*
1621 * Register, memory.
1622 */
1623 IEM_MC_BEGIN(2, 2);
1624 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1625 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1626 IEM_MC_ARG(uint64_t, uSrc, 1);
1627
1628 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1629 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1630 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1631 IEM_MC_PREPARE_SSE_USAGE();
1632
1633 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1634 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1635 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movddup, puDst, uSrc);
1636
1637 IEM_MC_ADVANCE_RIP();
1638 IEM_MC_END();
1639 }
1640 return VINF_SUCCESS;
1641}
1642
1643
1644/**
1645 * @opcode 0x13
1646 * @opcodesub !11 mr/reg
1647 * @oppfx none
1648 * @opcpuid sse
1649 * @opgroup og_sse_simdfp_datamove
1650 * @opxcpttype 5
1651 * @optest op1=1 op2=2 -> op1=2
1652 * @optest op1=0 op2=-42 -> op1=-42
1653 */
1654FNIEMOP_DEF(iemOp_movlps_Mq_Vq)
1655{
1656 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1657 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
1658 {
1659 IEMOP_MNEMONIC2(MR_MEM, MOVLPS, movlps, Mq_WO, Vq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1660
1661 IEM_MC_BEGIN(0, 2);
1662 IEM_MC_LOCAL(uint64_t, uSrc);
1663 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1664
1665 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1666 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1667 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1668 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1669
1670 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1671 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1672
1673 IEM_MC_ADVANCE_RIP();
1674 IEM_MC_END();
1675 return VINF_SUCCESS;
1676 }
1677
1678 /**
1679 * @opdone
1680 * @opmnemonic ud0f13m3
1681 * @opcode 0x13
1682 * @opcodesub 11 mr/reg
1683 * @oppfx none
1684 * @opunused immediate
1685 * @opcpuid sse
1686 * @optest ->
1687 */
1688 return IEMOP_RAISE_INVALID_OPCODE();
1689}
1690
1691
1692/**
1693 * @opcode 0x13
1694 * @opcodesub !11 mr/reg
1695 * @oppfx 0x66
1696 * @opcpuid sse2
1697 * @opgroup og_sse2_pcksclr_datamove
1698 * @opxcpttype 5
1699 * @optest op1=1 op2=2 -> op1=2
1700 * @optest op1=0 op2=-42 -> op1=-42
1701 */
1702FNIEMOP_DEF(iemOp_movlpd_Mq_Vq)
1703{
1704 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1705 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
1706 {
1707 IEMOP_MNEMONIC2(MR_MEM, MOVLPD, movlpd, Mq_WO, Vq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1708 IEM_MC_BEGIN(0, 2);
1709 IEM_MC_LOCAL(uint64_t, uSrc);
1710 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1711
1712 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1713 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1714 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1715 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
1716
1717 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1718 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
1719
1720 IEM_MC_ADVANCE_RIP();
1721 IEM_MC_END();
1722 return VINF_SUCCESS;
1723 }
1724
1725 /**
1726 * @opdone
1727 * @opmnemonic ud660f13m3
1728 * @opcode 0x13
1729 * @opcodesub 11 mr/reg
1730 * @oppfx 0x66
1731 * @opunused immediate
1732 * @opcpuid sse
1733 * @optest ->
1734 */
1735 return IEMOP_RAISE_INVALID_OPCODE();
1736}
1737
1738
1739/**
1740 * @opmnemonic udf30f13
1741 * @opcode 0x13
1742 * @oppfx 0xf3
1743 * @opunused intel-modrm
1744 * @opcpuid sse
1745 * @optest ->
1746 * @opdone
1747 */
1748
1749/**
1750 * @opmnemonic udf20f13
1751 * @opcode 0x13
1752 * @oppfx 0xf2
1753 * @opunused intel-modrm
1754 * @opcpuid sse
1755 * @optest ->
1756 * @opdone
1757 */
1758
1759/** Opcode 0x0f 0x14 - unpcklps Vx, Wx*/
1760FNIEMOP_STUB(iemOp_unpcklps_Vx_Wx);
1761/** Opcode 0x66 0x0f 0x14 - unpcklpd Vx, Wx */
1762FNIEMOP_STUB(iemOp_unpcklpd_Vx_Wx);
1763
1764/**
1765 * @opdone
1766 * @opmnemonic udf30f14
1767 * @opcode 0x14
1768 * @oppfx 0xf3
1769 * @opunused intel-modrm
1770 * @opcpuid sse
1771 * @optest ->
1772 * @opdone
1773 */
1774
1775/**
1776 * @opmnemonic udf20f14
1777 * @opcode 0x14
1778 * @oppfx 0xf2
1779 * @opunused intel-modrm
1780 * @opcpuid sse
1781 * @optest ->
1782 * @opdone
1783 */
1784
1785/** Opcode 0x0f 0x15 - unpckhps Vx, Wx */
1786FNIEMOP_STUB(iemOp_unpckhps_Vx_Wx);
1787/** Opcode 0x66 0x0f 0x15 - unpckhpd Vx, Wx */
1788FNIEMOP_STUB(iemOp_unpckhpd_Vx_Wx);
1789/* Opcode 0xf3 0x0f 0x15 - invalid */
1790/* Opcode 0xf2 0x0f 0x15 - invalid */
1791
1792/**
1793 * @opdone
1794 * @opmnemonic udf30f15
1795 * @opcode 0x15
1796 * @oppfx 0xf3
1797 * @opunused intel-modrm
1798 * @opcpuid sse
1799 * @optest ->
1800 * @opdone
1801 */
1802
1803/**
1804 * @opmnemonic udf20f15
1805 * @opcode 0x15
1806 * @oppfx 0xf2
1807 * @opunused intel-modrm
1808 * @opcpuid sse
1809 * @optest ->
1810 * @opdone
1811 */
1812
1813FNIEMOP_DEF(iemOp_movhps_Vdq_Mq__movlhps_Vdq_Uq)
1814{
1815 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1816 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1817 {
1818 /**
1819 * @opcode 0x16
1820 * @opcodesub 11 mr/reg
1821 * @oppfx none
1822 * @opcpuid sse
1823 * @opgroup og_sse_simdfp_datamove
1824 * @opxcpttype 5
1825 * @optest op1=1 op2=2 -> op1=2
1826 * @optest op1=0 op2=-42 -> op1=-42
1827 */
1828 IEMOP_MNEMONIC2(RM_REG, MOVLHPS, movlhps, VqHi_WO, Uq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1829
1830 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1831 IEM_MC_BEGIN(0, 1);
1832 IEM_MC_LOCAL(uint64_t, uSrc);
1833
1834 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1835 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1836 IEM_MC_FETCH_XREG_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1837 IEM_MC_STORE_XREG_HI_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1838
1839 IEM_MC_ADVANCE_RIP();
1840 IEM_MC_END();
1841 }
1842 else
1843 {
1844 /**
1845 * @opdone
1846 * @opcode 0x16
1847 * @opcodesub !11 mr/reg
1848 * @oppfx none
1849 * @opcpuid sse
1850 * @opgroup og_sse_simdfp_datamove
1851 * @opxcpttype 5
1852 * @optest op1=1 op2=2 -> op1=2
1853 * @optest op1=0 op2=-42 -> op1=-42
1854 * @opfunction iemOp_movhps_Vdq_Mq__movlhps_Vdq_Uq
1855 */
1856 IEMOP_MNEMONIC2(RM_MEM, MOVHPS, movhps, VqHi_WO, Mq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1857
1858 IEM_MC_BEGIN(0, 2);
1859 IEM_MC_LOCAL(uint64_t, uSrc);
1860 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1861
1862 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1863 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1864 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
1865 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1866
1867 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1868 IEM_MC_STORE_XREG_HI_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1869
1870 IEM_MC_ADVANCE_RIP();
1871 IEM_MC_END();
1872 }
1873 return VINF_SUCCESS;
1874}
1875
1876
1877/**
1878 * @opcode 0x16
1879 * @opcodesub !11 mr/reg
1880 * @oppfx 0x66
1881 * @opcpuid sse2
1882 * @opgroup og_sse2_pcksclr_datamove
1883 * @opxcpttype 5
1884 * @optest op1=1 op2=2 -> op1=2
1885 * @optest op1=0 op2=-42 -> op1=-42
1886 */
1887FNIEMOP_DEF(iemOp_movhpd_Vdq_Mq)
1888{
1889 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1890 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
1891 {
1892 IEMOP_MNEMONIC2(RM_MEM, MOVHPD, movhpd, VqHi_WO, Mq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1893 IEM_MC_BEGIN(0, 2);
1894 IEM_MC_LOCAL(uint64_t, uSrc);
1895 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1896
1897 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1898 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1899 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
1900 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
1901
1902 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1903 IEM_MC_STORE_XREG_HI_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
1904
1905 IEM_MC_ADVANCE_RIP();
1906 IEM_MC_END();
1907 return VINF_SUCCESS;
1908 }
1909
1910 /**
1911 * @opdone
1912 * @opmnemonic ud660f16m3
1913 * @opcode 0x16
1914 * @opcodesub 11 mr/reg
1915 * @oppfx 0x66
1916 * @opunused immediate
1917 * @opcpuid sse
1918 * @optest ->
1919 */
1920 return IEMOP_RAISE_INVALID_OPCODE();
1921}
1922
1923
1924/**
1925 * @opcode 0x16
1926 * @oppfx 0xf3
1927 * @opcpuid sse3
1928 * @opgroup og_sse3_pcksclr_datamove
1929 * @opxcpttype 4
1930 * @optest op1=-1 op2=0x00000002dddddddd00000001eeeeeeee ->
1931 * op1=0x00000002000000020000000100000001
1932 */
1933FNIEMOP_DEF(iemOp_movshdup_Vdq_Wdq)
1934{
1935 IEMOP_MNEMONIC2(RM, MOVSHDUP, movshdup, Vdq_WO, Wdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
1936 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
1937 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
1938 {
1939 /*
1940 * Register, register.
1941 */
1942 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1943 IEM_MC_BEGIN(2, 0);
1944 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1945 IEM_MC_ARG(PCRTUINT128U, puSrc, 1);
1946
1947 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1948 IEM_MC_PREPARE_SSE_USAGE();
1949
1950 IEM_MC_REF_XREG_U128_CONST(puSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
1951 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1952 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movshdup, puDst, puSrc);
1953
1954 IEM_MC_ADVANCE_RIP();
1955 IEM_MC_END();
1956 }
1957 else
1958 {
1959 /*
1960 * Register, memory.
1961 */
1962 IEM_MC_BEGIN(2, 2);
1963 IEM_MC_LOCAL(RTUINT128U, uSrc);
1964 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
1965 IEM_MC_ARG(PRTUINT128U, puDst, 0);
1966 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, puSrc, uSrc, 1);
1967
1968 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
1969 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
1970 IEM_MC_MAYBE_RAISE_SSE3_RELATED_XCPT();
1971 IEM_MC_PREPARE_SSE_USAGE();
1972
1973 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
1974 IEM_MC_REF_XREG_U128(puDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
1975 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_movshdup, puDst, puSrc);
1976
1977 IEM_MC_ADVANCE_RIP();
1978 IEM_MC_END();
1979 }
1980 return VINF_SUCCESS;
1981}
1982
1983/**
1984 * @opdone
1985 * @opmnemonic udf30f16
1986 * @opcode 0x16
1987 * @oppfx 0xf2
1988 * @opunused intel-modrm
1989 * @opcpuid sse
1990 * @optest ->
1991 * @opdone
1992 */
1993
1994
1995/**
1996 * @opcode 0x17
1997 * @opcodesub !11 mr/reg
1998 * @oppfx none
1999 * @opcpuid sse
2000 * @opgroup og_sse_simdfp_datamove
2001 * @opxcpttype 5
2002 * @optest op1=1 op2=2 -> op1=2
2003 * @optest op1=0 op2=-42 -> op1=-42
2004 */
2005FNIEMOP_DEF(iemOp_movhps_Mq_Vq)
2006{
2007 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2008 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
2009 {
2010 IEMOP_MNEMONIC2(MR_MEM, MOVHPS, movhps, Mq_WO, VqHi, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2011
2012 IEM_MC_BEGIN(0, 2);
2013 IEM_MC_LOCAL(uint64_t, uSrc);
2014 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2015
2016 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2017 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2018 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2019 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
2020
2021 IEM_MC_FETCH_XREG_HI_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2022 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2023
2024 IEM_MC_ADVANCE_RIP();
2025 IEM_MC_END();
2026 return VINF_SUCCESS;
2027 }
2028
2029 /**
2030 * @opdone
2031 * @opmnemonic ud0f17m3
2032 * @opcode 0x17
2033 * @opcodesub 11 mr/reg
2034 * @oppfx none
2035 * @opunused immediate
2036 * @opcpuid sse
2037 * @optest ->
2038 */
2039 return IEMOP_RAISE_INVALID_OPCODE();
2040}
2041
2042
2043/**
2044 * @opcode 0x17
2045 * @opcodesub !11 mr/reg
2046 * @oppfx 0x66
2047 * @opcpuid sse2
2048 * @opgroup og_sse2_pcksclr_datamove
2049 * @opxcpttype 5
2050 * @optest op1=1 op2=2 -> op1=2
2051 * @optest op1=0 op2=-42 -> op1=-42
2052 */
2053FNIEMOP_DEF(iemOp_movhpd_Mq_Vq)
2054{
2055 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2056 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
2057 {
2058 IEMOP_MNEMONIC2(MR_MEM, MOVHPD, movhpd, Mq_WO, VqHi, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2059
2060 IEM_MC_BEGIN(0, 2);
2061 IEM_MC_LOCAL(uint64_t, uSrc);
2062 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2063
2064 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2065 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2066 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2067 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
2068
2069 IEM_MC_FETCH_XREG_HI_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2070 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2071
2072 IEM_MC_ADVANCE_RIP();
2073 IEM_MC_END();
2074 return VINF_SUCCESS;
2075 }
2076
2077 /**
2078 * @opdone
2079 * @opmnemonic ud660f17m3
2080 * @opcode 0x17
2081 * @opcodesub 11 mr/reg
2082 * @oppfx 0x66
2083 * @opunused immediate
2084 * @opcpuid sse
2085 * @optest ->
2086 */
2087 return IEMOP_RAISE_INVALID_OPCODE();
2088}
2089
2090
2091/**
2092 * @opdone
2093 * @opmnemonic udf30f17
2094 * @opcode 0x17
2095 * @oppfx 0xf3
2096 * @opunused intel-modrm
2097 * @opcpuid sse
2098 * @optest ->
2099 * @opdone
2100 */
2101
2102/**
2103 * @opmnemonic udf20f17
2104 * @opcode 0x17
2105 * @oppfx 0xf2
2106 * @opunused intel-modrm
2107 * @opcpuid sse
2108 * @optest ->
2109 * @opdone
2110 */
2111
2112
2113/** Opcode 0x0f 0x18. */
2114FNIEMOP_DEF(iemOp_prefetch_Grp16)
2115{
2116 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2117 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
2118 {
2119 switch ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK)
2120 {
2121 case 4: /* Aliased to /0 for the time being according to AMD. */
2122 case 5: /* Aliased to /0 for the time being according to AMD. */
2123 case 6: /* Aliased to /0 for the time being according to AMD. */
2124 case 7: /* Aliased to /0 for the time being according to AMD. */
2125 case 0: IEMOP_MNEMONIC(prefetchNTA, "prefetchNTA m8"); break;
2126 case 1: IEMOP_MNEMONIC(prefetchT0, "prefetchT0 m8"); break;
2127 case 2: IEMOP_MNEMONIC(prefetchT1, "prefetchT1 m8"); break;
2128 case 3: IEMOP_MNEMONIC(prefetchT2, "prefetchT2 m8"); break;
2129 IEM_NOT_REACHED_DEFAULT_CASE_RET();
2130 }
2131
2132 IEM_MC_BEGIN(0, 1);
2133 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2134 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2135 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2136 /* Currently a NOP. */
2137 NOREF(GCPtrEffSrc);
2138 IEM_MC_ADVANCE_RIP();
2139 IEM_MC_END();
2140 return VINF_SUCCESS;
2141 }
2142
2143 return IEMOP_RAISE_INVALID_OPCODE();
2144}
2145
2146
2147/** Opcode 0x0f 0x19..0x1f. */
2148FNIEMOP_DEF(iemOp_nop_Ev)
2149{
2150 IEMOP_MNEMONIC(nop_Ev, "nop Ev");
2151 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2152 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
2153 {
2154 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2155 IEM_MC_BEGIN(0, 0);
2156 IEM_MC_ADVANCE_RIP();
2157 IEM_MC_END();
2158 }
2159 else
2160 {
2161 IEM_MC_BEGIN(0, 1);
2162 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2163 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2164 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2165 /* Currently a NOP. */
2166 NOREF(GCPtrEffSrc);
2167 IEM_MC_ADVANCE_RIP();
2168 IEM_MC_END();
2169 }
2170 return VINF_SUCCESS;
2171}
2172
2173
2174/** Opcode 0x0f 0x20. */
2175FNIEMOP_DEF(iemOp_mov_Rd_Cd)
2176{
2177 /* mod is ignored, as is operand size overrides. */
2178 IEMOP_MNEMONIC(mov_Rd_Cd, "mov Rd,Cd");
2179 IEMOP_HLP_MIN_386();
2180 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
2181 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize = IEMMODE_64BIT;
2182 else
2183 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize = IEMMODE_32BIT;
2184
2185 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2186 uint8_t iCrReg = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
2187 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK)
2188 {
2189 /* The lock prefix can be used to encode CR8 accesses on some CPUs. */
2190 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fMovCr8In32Bit)
2191 return IEMOP_RAISE_INVALID_OPCODE(); /* #UD takes precedence over #GP(), see test. */
2192 iCrReg |= 8;
2193 }
2194 switch (iCrReg)
2195 {
2196 case 0: case 2: case 3: case 4: case 8:
2197 break;
2198 default:
2199 return IEMOP_RAISE_INVALID_OPCODE();
2200 }
2201 IEMOP_HLP_DONE_DECODING();
2202
2203 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_mov_Rd_Cd, (X86_MODRM_RM_MASK & bRm) | pVCpu->iem.s.uRexB, iCrReg);
2204}
2205
2206
2207/** Opcode 0x0f 0x21. */
2208FNIEMOP_DEF(iemOp_mov_Rd_Dd)
2209{
2210 IEMOP_MNEMONIC(mov_Rd_Dd, "mov Rd,Dd");
2211 IEMOP_HLP_MIN_386();
2212 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2213 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2214 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_REX_R)
2215 return IEMOP_RAISE_INVALID_OPCODE();
2216 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_mov_Rd_Dd,
2217 (X86_MODRM_RM_MASK & bRm) | pVCpu->iem.s.uRexB,
2218 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK));
2219}
2220
2221
2222/** Opcode 0x0f 0x22. */
2223FNIEMOP_DEF(iemOp_mov_Cd_Rd)
2224{
2225 /* mod is ignored, as is operand size overrides. */
2226 IEMOP_MNEMONIC(mov_Cd_Rd, "mov Cd,Rd");
2227 IEMOP_HLP_MIN_386();
2228 if (pVCpu->iem.s.enmCpuMode == IEMMODE_64BIT)
2229 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize = IEMMODE_64BIT;
2230 else
2231 pVCpu->iem.s.enmEffOpSize = pVCpu->iem.s.enmDefOpSize = IEMMODE_32BIT;
2232
2233 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2234 uint8_t iCrReg = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
2235 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK)
2236 {
2237 /* The lock prefix can be used to encode CR8 accesses on some CPUs. */
2238 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fMovCr8In32Bit)
2239 return IEMOP_RAISE_INVALID_OPCODE(); /* #UD takes precedence over #GP(), see test. */
2240 iCrReg |= 8;
2241 }
2242 switch (iCrReg)
2243 {
2244 case 0: case 2: case 3: case 4: case 8:
2245 break;
2246 default:
2247 return IEMOP_RAISE_INVALID_OPCODE();
2248 }
2249 IEMOP_HLP_DONE_DECODING();
2250
2251 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_mov_Cd_Rd, iCrReg, (X86_MODRM_RM_MASK & bRm) | pVCpu->iem.s.uRexB);
2252}
2253
2254
2255/** Opcode 0x0f 0x23. */
2256FNIEMOP_DEF(iemOp_mov_Dd_Rd)
2257{
2258 IEMOP_MNEMONIC(mov_Dd_Rd, "mov Dd,Rd");
2259 IEMOP_HLP_MIN_386();
2260 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2261 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2262 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_REX_R)
2263 return IEMOP_RAISE_INVALID_OPCODE();
2264 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_mov_Dd_Rd,
2265 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK),
2266 (X86_MODRM_RM_MASK & bRm) | pVCpu->iem.s.uRexB);
2267}
2268
2269
2270/** Opcode 0x0f 0x24. */
2271FNIEMOP_DEF(iemOp_mov_Rd_Td)
2272{
2273 IEMOP_MNEMONIC(mov_Rd_Td, "mov Rd,Td");
2274 /** @todo works on 386 and 486. */
2275 /* The RM byte is not considered, see testcase. */
2276 return IEMOP_RAISE_INVALID_OPCODE();
2277}
2278
2279
2280/** Opcode 0x0f 0x26. */
2281FNIEMOP_DEF(iemOp_mov_Td_Rd)
2282{
2283 IEMOP_MNEMONIC(mov_Td_Rd, "mov Td,Rd");
2284 /** @todo works on 386 and 486. */
2285 /* The RM byte is not considered, see testcase. */
2286 return IEMOP_RAISE_INVALID_OPCODE();
2287}
2288
2289
2290/**
2291 * @opcode 0x28
2292 * @oppfx none
2293 * @opcpuid sse
2294 * @opgroup og_sse_simdfp_datamove
2295 * @opxcpttype 1
2296 * @optest op1=1 op2=2 -> op1=2
2297 * @optest op1=0 op2=-42 -> op1=-42
2298 */
2299FNIEMOP_DEF(iemOp_movaps_Vps_Wps)
2300{
2301 IEMOP_MNEMONIC2(RM, MOVAPS, movaps, Vps_WO, Wps, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2302 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2303 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
2304 {
2305 /*
2306 * Register, register.
2307 */
2308 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2309 IEM_MC_BEGIN(0, 0);
2310 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2311 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2312 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
2313 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
2314 IEM_MC_ADVANCE_RIP();
2315 IEM_MC_END();
2316 }
2317 else
2318 {
2319 /*
2320 * Register, memory.
2321 */
2322 IEM_MC_BEGIN(0, 2);
2323 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2324 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2325
2326 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2327 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2328 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2329 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2330
2331 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
2332 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
2333
2334 IEM_MC_ADVANCE_RIP();
2335 IEM_MC_END();
2336 }
2337 return VINF_SUCCESS;
2338}
2339
2340/**
2341 * @opcode 0x28
2342 * @oppfx 66
2343 * @opcpuid sse2
2344 * @opgroup og_sse2_pcksclr_datamove
2345 * @opxcpttype 1
2346 * @optest op1=1 op2=2 -> op1=2
2347 * @optest op1=0 op2=-42 -> op1=-42
2348 */
2349FNIEMOP_DEF(iemOp_movapd_Vpd_Wpd)
2350{
2351 IEMOP_MNEMONIC2(RM, MOVAPD, movapd, Vpd_WO, Wpd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2352 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2353 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
2354 {
2355 /*
2356 * Register, register.
2357 */
2358 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2359 IEM_MC_BEGIN(0, 0);
2360 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
2361 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2362 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
2363 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
2364 IEM_MC_ADVANCE_RIP();
2365 IEM_MC_END();
2366 }
2367 else
2368 {
2369 /*
2370 * Register, memory.
2371 */
2372 IEM_MC_BEGIN(0, 2);
2373 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2374 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2375
2376 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2377 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2378 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
2379 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2380
2381 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
2382 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
2383
2384 IEM_MC_ADVANCE_RIP();
2385 IEM_MC_END();
2386 }
2387 return VINF_SUCCESS;
2388}
2389
2390/* Opcode 0xf3 0x0f 0x28 - invalid */
2391/* Opcode 0xf2 0x0f 0x28 - invalid */
2392
2393/**
2394 * @opcode 0x29
2395 * @oppfx none
2396 * @opcpuid sse
2397 * @opgroup og_sse_simdfp_datamove
2398 * @opxcpttype 1
2399 * @optest op1=1 op2=2 -> op1=2
2400 * @optest op1=0 op2=-42 -> op1=-42
2401 */
2402FNIEMOP_DEF(iemOp_movaps_Wps_Vps)
2403{
2404 IEMOP_MNEMONIC2(MR, MOVAPS, movaps, Wps_WO, Vps, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2405 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2406 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
2407 {
2408 /*
2409 * Register, register.
2410 */
2411 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2412 IEM_MC_BEGIN(0, 0);
2413 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2414 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2415 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
2416 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2417 IEM_MC_ADVANCE_RIP();
2418 IEM_MC_END();
2419 }
2420 else
2421 {
2422 /*
2423 * Memory, register.
2424 */
2425 IEM_MC_BEGIN(0, 2);
2426 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2427 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2428
2429 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2430 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2431 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2432 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
2433
2434 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2435 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2436
2437 IEM_MC_ADVANCE_RIP();
2438 IEM_MC_END();
2439 }
2440 return VINF_SUCCESS;
2441}
2442
2443/**
2444 * @opcode 0x29
2445 * @oppfx 66
2446 * @opcpuid sse2
2447 * @opgroup og_sse2_pcksclr_datamove
2448 * @opxcpttype 1
2449 * @optest op1=1 op2=2 -> op1=2
2450 * @optest op1=0 op2=-42 -> op1=-42
2451 */
2452FNIEMOP_DEF(iemOp_movapd_Wpd_Vpd)
2453{
2454 IEMOP_MNEMONIC2(MR, MOVAPD, movapd, Wpd_WO, Vpd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2455 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2456 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
2457 {
2458 /*
2459 * Register, register.
2460 */
2461 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2462 IEM_MC_BEGIN(0, 0);
2463 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
2464 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2465 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
2466 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2467 IEM_MC_ADVANCE_RIP();
2468 IEM_MC_END();
2469 }
2470 else
2471 {
2472 /*
2473 * Memory, register.
2474 */
2475 IEM_MC_BEGIN(0, 2);
2476 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2477 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2478
2479 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2480 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2481 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
2482 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
2483
2484 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2485 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2486
2487 IEM_MC_ADVANCE_RIP();
2488 IEM_MC_END();
2489 }
2490 return VINF_SUCCESS;
2491}
2492
2493/* Opcode 0xf3 0x0f 0x29 - invalid */
2494/* Opcode 0xf2 0x0f 0x29 - invalid */
2495
2496
2497/** Opcode 0x0f 0x2a - cvtpi2ps Vps, Qpi */
2498FNIEMOP_STUB(iemOp_cvtpi2ps_Vps_Qpi); //NEXT
2499/** Opcode 0x66 0x0f 0x2a - cvtpi2pd Vpd, Qpi */
2500FNIEMOP_STUB(iemOp_cvtpi2pd_Vpd_Qpi); //NEXT
2501/** Opcode 0xf3 0x0f 0x2a - vcvtsi2ss Vss, Hss, Ey */
2502FNIEMOP_STUB(iemOp_cvtsi2ss_Vss_Ey); //NEXT
2503/** Opcode 0xf2 0x0f 0x2a - vcvtsi2sd Vsd, Hsd, Ey */
2504FNIEMOP_STUB(iemOp_cvtsi2sd_Vsd_Ey); //NEXT
2505
2506
2507/**
2508 * @opcode 0x2b
2509 * @opcodesub !11 mr/reg
2510 * @oppfx none
2511 * @opcpuid sse
2512 * @opgroup og_sse1_cachect
2513 * @opxcpttype 1
2514 * @optest op1=1 op2=2 -> op1=2
2515 * @optest op1=0 op2=-42 -> op1=-42
2516 */
2517FNIEMOP_DEF(iemOp_movntps_Mps_Vps)
2518{
2519 IEMOP_MNEMONIC2(MR_MEM, MOVNTPS, movntps, Mps_WO, Vps, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2520 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2521 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
2522 {
2523 /*
2524 * memory, register.
2525 */
2526 IEM_MC_BEGIN(0, 2);
2527 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2528 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2529
2530 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2531 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2532 IEM_MC_MAYBE_RAISE_SSE_RELATED_XCPT();
2533 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2534
2535 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2536 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2537
2538 IEM_MC_ADVANCE_RIP();
2539 IEM_MC_END();
2540 }
2541 /* The register, register encoding is invalid. */
2542 else
2543 return IEMOP_RAISE_INVALID_OPCODE();
2544 return VINF_SUCCESS;
2545}
2546
2547/**
2548 * @opcode 0x2b
2549 * @opcodesub !11 mr/reg
2550 * @oppfx 0x66
2551 * @opcpuid sse2
2552 * @opgroup og_sse2_cachect
2553 * @opxcpttype 1
2554 * @optest op1=1 op2=2 -> op1=2
2555 * @optest op1=0 op2=-42 -> op1=-42
2556 */
2557FNIEMOP_DEF(iemOp_movntpd_Mpd_Vpd)
2558{
2559 IEMOP_MNEMONIC2(MR_MEM, MOVNTPD, movntpd, Mpd_WO, Vpd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
2560 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
2561 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
2562 {
2563 /*
2564 * memory, register.
2565 */
2566 IEM_MC_BEGIN(0, 2);
2567 IEM_MC_LOCAL(RTUINT128U, uSrc); /** @todo optimize this one day... */
2568 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
2569
2570 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
2571 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2572 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
2573 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
2574
2575 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
2576 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
2577
2578 IEM_MC_ADVANCE_RIP();
2579 IEM_MC_END();
2580 }
2581 /* The register, register encoding is invalid. */
2582 else
2583 return IEMOP_RAISE_INVALID_OPCODE();
2584 return VINF_SUCCESS;
2585}
2586/* Opcode 0xf3 0x0f 0x2b - invalid */
2587/* Opcode 0xf2 0x0f 0x2b - invalid */
2588
2589
2590/** Opcode 0x0f 0x2c - cvttps2pi Ppi, Wps */
2591FNIEMOP_STUB(iemOp_cvttps2pi_Ppi_Wps);
2592/** Opcode 0x66 0x0f 0x2c - cvttpd2pi Ppi, Wpd */
2593FNIEMOP_STUB(iemOp_cvttpd2pi_Ppi_Wpd);
2594/** Opcode 0xf3 0x0f 0x2c - cvttss2si Gy, Wss */
2595FNIEMOP_STUB(iemOp_cvttss2si_Gy_Wss);
2596/** Opcode 0xf2 0x0f 0x2c - cvttsd2si Gy, Wsd */
2597FNIEMOP_STUB(iemOp_cvttsd2si_Gy_Wsd);
2598
2599/** Opcode 0x0f 0x2d - cvtps2pi Ppi, Wps */
2600FNIEMOP_STUB(iemOp_cvtps2pi_Ppi_Wps);
2601/** Opcode 0x66 0x0f 0x2d - cvtpd2pi Qpi, Wpd */
2602FNIEMOP_STUB(iemOp_cvtpd2pi_Qpi_Wpd);
2603/** Opcode 0xf3 0x0f 0x2d - cvtss2si Gy, Wss */
2604FNIEMOP_STUB(iemOp_cvtss2si_Gy_Wss);
2605/** Opcode 0xf2 0x0f 0x2d - cvtsd2si Gy, Wsd */
2606FNIEMOP_STUB(iemOp_cvtsd2si_Gy_Wsd);
2607
2608/** Opcode 0x0f 0x2e - ucomiss Vss, Wss */
2609FNIEMOP_STUB(iemOp_ucomiss_Vss_Wss); // NEXT
2610/** Opcode 0x66 0x0f 0x2e - ucomisd Vsd, Wsd */
2611FNIEMOP_STUB(iemOp_ucomisd_Vsd_Wsd); // NEXT
2612/* Opcode 0xf3 0x0f 0x2e - invalid */
2613/* Opcode 0xf2 0x0f 0x2e - invalid */
2614
2615/** Opcode 0x0f 0x2f - comiss Vss, Wss */
2616FNIEMOP_STUB(iemOp_comiss_Vss_Wss);
2617/** Opcode 0x66 0x0f 0x2f - comisd Vsd, Wsd */
2618FNIEMOP_STUB(iemOp_comisd_Vsd_Wsd);
2619/* Opcode 0xf3 0x0f 0x2f - invalid */
2620/* Opcode 0xf2 0x0f 0x2f - invalid */
2621
2622/** Opcode 0x0f 0x30. */
2623FNIEMOP_DEF(iemOp_wrmsr)
2624{
2625 IEMOP_MNEMONIC(wrmsr, "wrmsr");
2626 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2627 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_wrmsr);
2628}
2629
2630
2631/** Opcode 0x0f 0x31. */
2632FNIEMOP_DEF(iemOp_rdtsc)
2633{
2634 IEMOP_MNEMONIC(rdtsc, "rdtsc");
2635 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2636 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_rdtsc);
2637}
2638
2639
2640/** Opcode 0x0f 0x33. */
2641FNIEMOP_DEF(iemOp_rdmsr)
2642{
2643 IEMOP_MNEMONIC(rdmsr, "rdmsr");
2644 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2645 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_rdmsr);
2646}
2647
2648
2649/** Opcode 0x0f 0x34. */
2650FNIEMOP_DEF(iemOp_rdpmc)
2651{
2652 IEMOP_MNEMONIC(rdpmc, "rdpmc");
2653 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
2654 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_rdpmc);
2655}
2656
2657
2658/** Opcode 0x0f 0x34. */
2659FNIEMOP_STUB(iemOp_sysenter);
2660/** Opcode 0x0f 0x35. */
2661FNIEMOP_STUB(iemOp_sysexit);
2662/** Opcode 0x0f 0x37. */
2663FNIEMOP_STUB(iemOp_getsec);
2664
2665
2666/** Opcode 0x0f 0x38. */
2667FNIEMOP_DEF(iemOp_3byte_Esc_0f_38)
2668{
2669#ifdef IEM_WITH_THREE_0F_38
2670 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
2671 return FNIEMOP_CALL(g_apfnThreeByte0f38[(uintptr_t)b * 4 + pVCpu->iem.s.idxPrefix]);
2672#else
2673 IEMOP_BITCH_ABOUT_STUB();
2674 return VERR_IEM_INSTR_NOT_IMPLEMENTED;
2675#endif
2676}
2677
2678
2679/** Opcode 0x0f 0x3a. */
2680FNIEMOP_DEF(iemOp_3byte_Esc_0f_3a)
2681{
2682#ifdef IEM_WITH_THREE_0F_3A
2683 uint8_t b; IEM_OPCODE_GET_NEXT_U8(&b);
2684 return FNIEMOP_CALL(g_apfnThreeByte0f3a[(uintptr_t)b * 4 + pVCpu->iem.s.idxPrefix]);
2685#else
2686 IEMOP_BITCH_ABOUT_STUB();
2687 return VERR_IEM_INSTR_NOT_IMPLEMENTED;
2688#endif
2689}
2690
2691
2692/**
2693 * Implements a conditional move.
2694 *
2695 * Wish there was an obvious way to do this where we could share and reduce
2696 * code bloat.
2697 *
2698 * @param a_Cnd The conditional "microcode" operation.
2699 */
2700#define CMOV_X(a_Cnd) \
2701 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm); \
2702 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT)) \
2703 { \
2704 switch (pVCpu->iem.s.enmEffOpSize) \
2705 { \
2706 case IEMMODE_16BIT: \
2707 IEM_MC_BEGIN(0, 1); \
2708 IEM_MC_LOCAL(uint16_t, u16Tmp); \
2709 a_Cnd { \
2710 IEM_MC_FETCH_GREG_U16(u16Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB); \
2711 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Tmp); \
2712 } IEM_MC_ENDIF(); \
2713 IEM_MC_ADVANCE_RIP(); \
2714 IEM_MC_END(); \
2715 return VINF_SUCCESS; \
2716 \
2717 case IEMMODE_32BIT: \
2718 IEM_MC_BEGIN(0, 1); \
2719 IEM_MC_LOCAL(uint32_t, u32Tmp); \
2720 a_Cnd { \
2721 IEM_MC_FETCH_GREG_U32(u32Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB); \
2722 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Tmp); \
2723 } IEM_MC_ELSE() { \
2724 IEM_MC_CLEAR_HIGH_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg); \
2725 } IEM_MC_ENDIF(); \
2726 IEM_MC_ADVANCE_RIP(); \
2727 IEM_MC_END(); \
2728 return VINF_SUCCESS; \
2729 \
2730 case IEMMODE_64BIT: \
2731 IEM_MC_BEGIN(0, 1); \
2732 IEM_MC_LOCAL(uint64_t, u64Tmp); \
2733 a_Cnd { \
2734 IEM_MC_FETCH_GREG_U64(u64Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB); \
2735 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Tmp); \
2736 } IEM_MC_ENDIF(); \
2737 IEM_MC_ADVANCE_RIP(); \
2738 IEM_MC_END(); \
2739 return VINF_SUCCESS; \
2740 \
2741 IEM_NOT_REACHED_DEFAULT_CASE_RET(); \
2742 } \
2743 } \
2744 else \
2745 { \
2746 switch (pVCpu->iem.s.enmEffOpSize) \
2747 { \
2748 case IEMMODE_16BIT: \
2749 IEM_MC_BEGIN(0, 2); \
2750 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc); \
2751 IEM_MC_LOCAL(uint16_t, u16Tmp); \
2752 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0); \
2753 IEM_MC_FETCH_MEM_U16(u16Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc); \
2754 a_Cnd { \
2755 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Tmp); \
2756 } IEM_MC_ENDIF(); \
2757 IEM_MC_ADVANCE_RIP(); \
2758 IEM_MC_END(); \
2759 return VINF_SUCCESS; \
2760 \
2761 case IEMMODE_32BIT: \
2762 IEM_MC_BEGIN(0, 2); \
2763 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc); \
2764 IEM_MC_LOCAL(uint32_t, u32Tmp); \
2765 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0); \
2766 IEM_MC_FETCH_MEM_U32(u32Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc); \
2767 a_Cnd { \
2768 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Tmp); \
2769 } IEM_MC_ELSE() { \
2770 IEM_MC_CLEAR_HIGH_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg); \
2771 } IEM_MC_ENDIF(); \
2772 IEM_MC_ADVANCE_RIP(); \
2773 IEM_MC_END(); \
2774 return VINF_SUCCESS; \
2775 \
2776 case IEMMODE_64BIT: \
2777 IEM_MC_BEGIN(0, 2); \
2778 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc); \
2779 IEM_MC_LOCAL(uint64_t, u64Tmp); \
2780 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0); \
2781 IEM_MC_FETCH_MEM_U64(u64Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc); \
2782 a_Cnd { \
2783 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Tmp); \
2784 } IEM_MC_ENDIF(); \
2785 IEM_MC_ADVANCE_RIP(); \
2786 IEM_MC_END(); \
2787 return VINF_SUCCESS; \
2788 \
2789 IEM_NOT_REACHED_DEFAULT_CASE_RET(); \
2790 } \
2791 } do {} while (0)
2792
2793
2794
2795/** Opcode 0x0f 0x40. */
2796FNIEMOP_DEF(iemOp_cmovo_Gv_Ev)
2797{
2798 IEMOP_MNEMONIC(cmovo_Gv_Ev, "cmovo Gv,Ev");
2799 CMOV_X(IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF));
2800}
2801
2802
2803/** Opcode 0x0f 0x41. */
2804FNIEMOP_DEF(iemOp_cmovno_Gv_Ev)
2805{
2806 IEMOP_MNEMONIC(cmovno_Gv_Ev, "cmovno Gv,Ev");
2807 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_OF));
2808}
2809
2810
2811/** Opcode 0x0f 0x42. */
2812FNIEMOP_DEF(iemOp_cmovc_Gv_Ev)
2813{
2814 IEMOP_MNEMONIC(cmovc_Gv_Ev, "cmovc Gv,Ev");
2815 CMOV_X(IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF));
2816}
2817
2818
2819/** Opcode 0x0f 0x43. */
2820FNIEMOP_DEF(iemOp_cmovnc_Gv_Ev)
2821{
2822 IEMOP_MNEMONIC(cmovnc_Gv_Ev, "cmovnc Gv,Ev");
2823 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_CF));
2824}
2825
2826
2827/** Opcode 0x0f 0x44. */
2828FNIEMOP_DEF(iemOp_cmove_Gv_Ev)
2829{
2830 IEMOP_MNEMONIC(cmove_Gv_Ev, "cmove Gv,Ev");
2831 CMOV_X(IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF));
2832}
2833
2834
2835/** Opcode 0x0f 0x45. */
2836FNIEMOP_DEF(iemOp_cmovne_Gv_Ev)
2837{
2838 IEMOP_MNEMONIC(cmovne_Gv_Ev, "cmovne Gv,Ev");
2839 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_ZF));
2840}
2841
2842
2843/** Opcode 0x0f 0x46. */
2844FNIEMOP_DEF(iemOp_cmovbe_Gv_Ev)
2845{
2846 IEMOP_MNEMONIC(cmovbe_Gv_Ev, "cmovbe Gv,Ev");
2847 CMOV_X(IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF));
2848}
2849
2850
2851/** Opcode 0x0f 0x47. */
2852FNIEMOP_DEF(iemOp_cmovnbe_Gv_Ev)
2853{
2854 IEMOP_MNEMONIC(cmovnbe_Gv_Ev, "cmovnbe Gv,Ev");
2855 CMOV_X(IEM_MC_IF_EFL_NO_BITS_SET(X86_EFL_CF | X86_EFL_ZF));
2856}
2857
2858
2859/** Opcode 0x0f 0x48. */
2860FNIEMOP_DEF(iemOp_cmovs_Gv_Ev)
2861{
2862 IEMOP_MNEMONIC(cmovs_Gv_Ev, "cmovs Gv,Ev");
2863 CMOV_X(IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF));
2864}
2865
2866
2867/** Opcode 0x0f 0x49. */
2868FNIEMOP_DEF(iemOp_cmovns_Gv_Ev)
2869{
2870 IEMOP_MNEMONIC(cmovns_Gv_Ev, "cmovns Gv,Ev");
2871 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_SF));
2872}
2873
2874
2875/** Opcode 0x0f 0x4a. */
2876FNIEMOP_DEF(iemOp_cmovp_Gv_Ev)
2877{
2878 IEMOP_MNEMONIC(cmovp_Gv_Ev, "cmovp Gv,Ev");
2879 CMOV_X(IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF));
2880}
2881
2882
2883/** Opcode 0x0f 0x4b. */
2884FNIEMOP_DEF(iemOp_cmovnp_Gv_Ev)
2885{
2886 IEMOP_MNEMONIC(cmovnp_Gv_Ev, "cmovnp Gv,Ev");
2887 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_PF));
2888}
2889
2890
2891/** Opcode 0x0f 0x4c. */
2892FNIEMOP_DEF(iemOp_cmovl_Gv_Ev)
2893{
2894 IEMOP_MNEMONIC(cmovl_Gv_Ev, "cmovl Gv,Ev");
2895 CMOV_X(IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF));
2896}
2897
2898
2899/** Opcode 0x0f 0x4d. */
2900FNIEMOP_DEF(iemOp_cmovnl_Gv_Ev)
2901{
2902 IEMOP_MNEMONIC(cmovnl_Gv_Ev, "cmovnl Gv,Ev");
2903 CMOV_X(IEM_MC_IF_EFL_BITS_EQ(X86_EFL_SF, X86_EFL_OF));
2904}
2905
2906
2907/** Opcode 0x0f 0x4e. */
2908FNIEMOP_DEF(iemOp_cmovle_Gv_Ev)
2909{
2910 IEMOP_MNEMONIC(cmovle_Gv_Ev, "cmovle Gv,Ev");
2911 CMOV_X(IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF));
2912}
2913
2914
2915/** Opcode 0x0f 0x4f. */
2916FNIEMOP_DEF(iemOp_cmovnle_Gv_Ev)
2917{
2918 IEMOP_MNEMONIC(cmovnle_Gv_Ev, "cmovnle Gv,Ev");
2919 CMOV_X(IEM_MC_IF_EFL_BIT_NOT_SET_AND_BITS_EQ(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF));
2920}
2921
2922#undef CMOV_X
2923
2924/** Opcode 0x0f 0x50 - movmskps Gy, Ups */
2925FNIEMOP_STUB(iemOp_movmskps_Gy_Ups);
2926/** Opcode 0x66 0x0f 0x50 - movmskpd Gy, Upd */
2927FNIEMOP_STUB(iemOp_movmskpd_Gy_Upd);
2928/* Opcode 0xf3 0x0f 0x50 - invalid */
2929/* Opcode 0xf2 0x0f 0x50 - invalid */
2930
2931/** Opcode 0x0f 0x51 - sqrtps Vps, Wps */
2932FNIEMOP_STUB(iemOp_sqrtps_Vps_Wps);
2933/** Opcode 0x66 0x0f 0x51 - sqrtpd Vpd, Wpd */
2934FNIEMOP_STUB(iemOp_sqrtpd_Vpd_Wpd);
2935/** Opcode 0xf3 0x0f 0x51 - sqrtss Vss, Wss */
2936FNIEMOP_STUB(iemOp_sqrtss_Vss_Wss);
2937/** Opcode 0xf2 0x0f 0x51 - sqrtsd Vsd, Wsd */
2938FNIEMOP_STUB(iemOp_sqrtsd_Vsd_Wsd);
2939
2940/** Opcode 0x0f 0x52 - rsqrtps Vps, Wps */
2941FNIEMOP_STUB(iemOp_rsqrtps_Vps_Wps);
2942/* Opcode 0x66 0x0f 0x52 - invalid */
2943/** Opcode 0xf3 0x0f 0x52 - rsqrtss Vss, Wss */
2944FNIEMOP_STUB(iemOp_rsqrtss_Vss_Wss);
2945/* Opcode 0xf2 0x0f 0x52 - invalid */
2946
2947/** Opcode 0x0f 0x53 - rcpps Vps, Wps */
2948FNIEMOP_STUB(iemOp_rcpps_Vps_Wps);
2949/* Opcode 0x66 0x0f 0x53 - invalid */
2950/** Opcode 0xf3 0x0f 0x53 - rcpss Vss, Wss */
2951FNIEMOP_STUB(iemOp_rcpss_Vss_Wss);
2952/* Opcode 0xf2 0x0f 0x53 - invalid */
2953
2954/** Opcode 0x0f 0x54 - andps Vps, Wps */
2955FNIEMOP_STUB(iemOp_andps_Vps_Wps);
2956/** Opcode 0x66 0x0f 0x54 - andpd Vpd, Wpd */
2957FNIEMOP_STUB(iemOp_andpd_Vpd_Wpd);
2958/* Opcode 0xf3 0x0f 0x54 - invalid */
2959/* Opcode 0xf2 0x0f 0x54 - invalid */
2960
2961/** Opcode 0x0f 0x55 - andnps Vps, Wps */
2962FNIEMOP_STUB(iemOp_andnps_Vps_Wps);
2963/** Opcode 0x66 0x0f 0x55 - andnpd Vpd, Wpd */
2964FNIEMOP_STUB(iemOp_andnpd_Vpd_Wpd);
2965/* Opcode 0xf3 0x0f 0x55 - invalid */
2966/* Opcode 0xf2 0x0f 0x55 - invalid */
2967
2968/** Opcode 0x0f 0x56 - orps Vps, Wps */
2969FNIEMOP_STUB(iemOp_orps_Vps_Wps);
2970/** Opcode 0x66 0x0f 0x56 - orpd Vpd, Wpd */
2971FNIEMOP_STUB(iemOp_orpd_Vpd_Wpd);
2972/* Opcode 0xf3 0x0f 0x56 - invalid */
2973/* Opcode 0xf2 0x0f 0x56 - invalid */
2974
2975/** Opcode 0x0f 0x57 - xorps Vps, Wps */
2976FNIEMOP_STUB(iemOp_xorps_Vps_Wps);
2977/** Opcode 0x66 0x0f 0x57 - xorpd Vpd, Wpd */
2978FNIEMOP_STUB(iemOp_xorpd_Vpd_Wpd);
2979/* Opcode 0xf3 0x0f 0x57 - invalid */
2980/* Opcode 0xf2 0x0f 0x57 - invalid */
2981
2982/** Opcode 0x0f 0x58 - addps Vps, Wps */
2983FNIEMOP_STUB(iemOp_addps_Vps_Wps);
2984/** Opcode 0x66 0x0f 0x58 - addpd Vpd, Wpd */
2985FNIEMOP_STUB(iemOp_addpd_Vpd_Wpd);
2986/** Opcode 0xf3 0x0f 0x58 - addss Vss, Wss */
2987FNIEMOP_STUB(iemOp_addss_Vss_Wss);
2988/** Opcode 0xf2 0x0f 0x58 - addsd Vsd, Wsd */
2989FNIEMOP_STUB(iemOp_addsd_Vsd_Wsd);
2990
2991/** Opcode 0x0f 0x59 - mulps Vps, Wps */
2992FNIEMOP_STUB(iemOp_mulps_Vps_Wps);
2993/** Opcode 0x66 0x0f 0x59 - mulpd Vpd, Wpd */
2994FNIEMOP_STUB(iemOp_mulpd_Vpd_Wpd);
2995/** Opcode 0xf3 0x0f 0x59 - mulss Vss, Wss */
2996FNIEMOP_STUB(iemOp_mulss_Vss_Wss);
2997/** Opcode 0xf2 0x0f 0x59 - mulsd Vsd, Wsd */
2998FNIEMOP_STUB(iemOp_mulsd_Vsd_Wsd);
2999
3000/** Opcode 0x0f 0x5a - cvtps2pd Vpd, Wps */
3001FNIEMOP_STUB(iemOp_cvtps2pd_Vpd_Wps);
3002/** Opcode 0x66 0x0f 0x5a - cvtpd2ps Vps, Wpd */
3003FNIEMOP_STUB(iemOp_cvtpd2ps_Vps_Wpd);
3004/** Opcode 0xf3 0x0f 0x5a - cvtss2sd Vsd, Wss */
3005FNIEMOP_STUB(iemOp_cvtss2sd_Vsd_Wss);
3006/** Opcode 0xf2 0x0f 0x5a - cvtsd2ss Vss, Wsd */
3007FNIEMOP_STUB(iemOp_cvtsd2ss_Vss_Wsd);
3008
3009/** Opcode 0x0f 0x5b - cvtdq2ps Vps, Wdq */
3010FNIEMOP_STUB(iemOp_cvtdq2ps_Vps_Wdq);
3011/** Opcode 0x66 0x0f 0x5b - cvtps2dq Vdq, Wps */
3012FNIEMOP_STUB(iemOp_cvtps2dq_Vdq_Wps);
3013/** Opcode 0xf3 0x0f 0x5b - cvttps2dq Vdq, Wps */
3014FNIEMOP_STUB(iemOp_cvttps2dq_Vdq_Wps);
3015/* Opcode 0xf2 0x0f 0x5b - invalid */
3016
3017/** Opcode 0x0f 0x5c - subps Vps, Wps */
3018FNIEMOP_STUB(iemOp_subps_Vps_Wps);
3019/** Opcode 0x66 0x0f 0x5c - subpd Vpd, Wpd */
3020FNIEMOP_STUB(iemOp_subpd_Vpd_Wpd);
3021/** Opcode 0xf3 0x0f 0x5c - subss Vss, Wss */
3022FNIEMOP_STUB(iemOp_subss_Vss_Wss);
3023/** Opcode 0xf2 0x0f 0x5c - subsd Vsd, Wsd */
3024FNIEMOP_STUB(iemOp_subsd_Vsd_Wsd);
3025
3026/** Opcode 0x0f 0x5d - minps Vps, Wps */
3027FNIEMOP_STUB(iemOp_minps_Vps_Wps);
3028/** Opcode 0x66 0x0f 0x5d - minpd Vpd, Wpd */
3029FNIEMOP_STUB(iemOp_minpd_Vpd_Wpd);
3030/** Opcode 0xf3 0x0f 0x5d - minss Vss, Wss */
3031FNIEMOP_STUB(iemOp_minss_Vss_Wss);
3032/** Opcode 0xf2 0x0f 0x5d - minsd Vsd, Wsd */
3033FNIEMOP_STUB(iemOp_minsd_Vsd_Wsd);
3034
3035/** Opcode 0x0f 0x5e - divps Vps, Wps */
3036FNIEMOP_STUB(iemOp_divps_Vps_Wps);
3037/** Opcode 0x66 0x0f 0x5e - divpd Vpd, Wpd */
3038FNIEMOP_STUB(iemOp_divpd_Vpd_Wpd);
3039/** Opcode 0xf3 0x0f 0x5e - divss Vss, Wss */
3040FNIEMOP_STUB(iemOp_divss_Vss_Wss);
3041/** Opcode 0xf2 0x0f 0x5e - divsd Vsd, Wsd */
3042FNIEMOP_STUB(iemOp_divsd_Vsd_Wsd);
3043
3044/** Opcode 0x0f 0x5f - maxps Vps, Wps */
3045FNIEMOP_STUB(iemOp_maxps_Vps_Wps);
3046/** Opcode 0x66 0x0f 0x5f - maxpd Vpd, Wpd */
3047FNIEMOP_STUB(iemOp_maxpd_Vpd_Wpd);
3048/** Opcode 0xf3 0x0f 0x5f - maxss Vss, Wss */
3049FNIEMOP_STUB(iemOp_maxss_Vss_Wss);
3050/** Opcode 0xf2 0x0f 0x5f - maxsd Vsd, Wsd */
3051FNIEMOP_STUB(iemOp_maxsd_Vsd_Wsd);
3052
3053/**
3054 * Common worker for MMX instructions on the forms:
3055 * pxxxx mm1, mm2/mem32
3056 *
3057 * The 2nd operand is the first half of a register, which in the memory case
3058 * means a 32-bit memory access for MMX and 128-bit aligned 64-bit or 128-bit
3059 * memory accessed for MMX.
3060 *
3061 * Exceptions type 4.
3062 */
3063FNIEMOP_DEF_1(iemOpCommonMmx_LowLow_To_Full, PCIEMOPMEDIAF1L1, pImpl)
3064{
3065 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3066 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3067 {
3068 /*
3069 * Register, register.
3070 */
3071 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3072 IEM_MC_BEGIN(2, 0);
3073 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3074 IEM_MC_ARG(uint64_t const *, pSrc, 1);
3075 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3076 IEM_MC_PREPARE_SSE_USAGE();
3077 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3078 IEM_MC_REF_XREG_U64_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3079 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
3080 IEM_MC_ADVANCE_RIP();
3081 IEM_MC_END();
3082 }
3083 else
3084 {
3085 /*
3086 * Register, memory.
3087 */
3088 IEM_MC_BEGIN(2, 2);
3089 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3090 IEM_MC_LOCAL(uint64_t, uSrc);
3091 IEM_MC_ARG_LOCAL_REF(uint64_t const *, pSrc, uSrc, 1);
3092 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3093
3094 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3095 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3096 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3097 IEM_MC_FETCH_MEM_U64_ALIGN_U128(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3098
3099 IEM_MC_PREPARE_SSE_USAGE();
3100 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3101 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
3102
3103 IEM_MC_ADVANCE_RIP();
3104 IEM_MC_END();
3105 }
3106 return VINF_SUCCESS;
3107}
3108
3109
3110/**
3111 * Common worker for SSE2 instructions on the forms:
3112 * pxxxx xmm1, xmm2/mem128
3113 *
3114 * The 2nd operand is the first half of a register, which in the memory case
3115 * means a 32-bit memory access for MMX and 128-bit aligned 64-bit or 128-bit
3116 * memory accessed for MMX.
3117 *
3118 * Exceptions type 4.
3119 */
3120FNIEMOP_DEF_1(iemOpCommonSse_LowLow_To_Full, PCIEMOPMEDIAF1L1, pImpl)
3121{
3122 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3123 if (!pImpl->pfnU64)
3124 return IEMOP_RAISE_INVALID_OPCODE();
3125 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3126 {
3127 /*
3128 * Register, register.
3129 */
3130 /** @todo testcase: REX.B / REX.R and MMX register indexing. Ignored? */
3131 /** @todo testcase: REX.B / REX.R and segment register indexing. Ignored? */
3132 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3133 IEM_MC_BEGIN(2, 0);
3134 IEM_MC_ARG(uint64_t *, pDst, 0);
3135 IEM_MC_ARG(uint32_t const *, pSrc, 1);
3136 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3137 IEM_MC_PREPARE_FPU_USAGE();
3138 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3139 IEM_MC_REF_MREG_U32_CONST(pSrc, bRm & X86_MODRM_RM_MASK);
3140 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
3141 IEM_MC_ADVANCE_RIP();
3142 IEM_MC_END();
3143 }
3144 else
3145 {
3146 /*
3147 * Register, memory.
3148 */
3149 IEM_MC_BEGIN(2, 2);
3150 IEM_MC_ARG(uint64_t *, pDst, 0);
3151 IEM_MC_LOCAL(uint32_t, uSrc);
3152 IEM_MC_ARG_LOCAL_REF(uint32_t const *, pSrc, uSrc, 1);
3153 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3154
3155 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3156 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3157 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3158 IEM_MC_FETCH_MEM_U32(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3159
3160 IEM_MC_PREPARE_FPU_USAGE();
3161 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3162 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
3163
3164 IEM_MC_ADVANCE_RIP();
3165 IEM_MC_END();
3166 }
3167 return VINF_SUCCESS;
3168}
3169
3170
3171/** Opcode 0x0f 0x60 - punpcklbw Pq, Qd */
3172FNIEMOP_DEF(iemOp_punpcklbw_Pq_Qd)
3173{
3174 IEMOP_MNEMONIC(punpcklbw, "punpcklbw Pq, Qd");
3175 return FNIEMOP_CALL_1(iemOpCommonMmx_LowLow_To_Full, &g_iemAImpl_punpcklbw);
3176}
3177
3178/** Opcode 0x66 0x0f 0x60 - punpcklbw Vx, W */
3179FNIEMOP_DEF(iemOp_punpcklbw_Vx_Wx)
3180{
3181 IEMOP_MNEMONIC(vpunpcklbw_Vx_Wx, "vpunpcklbw Vx, Wx");
3182 return FNIEMOP_CALL_1(iemOpCommonSse_LowLow_To_Full, &g_iemAImpl_punpcklbw);
3183}
3184
3185/* Opcode 0xf3 0x0f 0x60 - invalid */
3186
3187
3188/** Opcode 0x0f 0x61 - punpcklwd Pq, Qd */
3189FNIEMOP_DEF(iemOp_punpcklwd_Pq_Qd)
3190{
3191 IEMOP_MNEMONIC(punpcklwd, "punpcklwd Pq, Qd"); /** @todo AMD mark the MMX version as 3DNow!. Intel says MMX CPUID req. */
3192 return FNIEMOP_CALL_1(iemOpCommonMmx_LowLow_To_Full, &g_iemAImpl_punpcklwd);
3193}
3194
3195/** Opcode 0x66 0x0f 0x61 - punpcklwd Vx, Wx */
3196FNIEMOP_DEF(iemOp_punpcklwd_Vx_Wx)
3197{
3198 IEMOP_MNEMONIC(vpunpcklwd_Vx_Wx, "punpcklwd Vx, Wx");
3199 return FNIEMOP_CALL_1(iemOpCommonSse_LowLow_To_Full, &g_iemAImpl_punpcklwd);
3200}
3201
3202/* Opcode 0xf3 0x0f 0x61 - invalid */
3203
3204
3205/** Opcode 0x0f 0x62 - punpckldq Pq, Qd */
3206FNIEMOP_DEF(iemOp_punpckldq_Pq_Qd)
3207{
3208 IEMOP_MNEMONIC(punpckldq, "punpckldq Pq, Qd");
3209 return FNIEMOP_CALL_1(iemOpCommonMmx_LowLow_To_Full, &g_iemAImpl_punpckldq);
3210}
3211
3212/** Opcode 0x66 0x0f 0x62 - punpckldq Vx, Wx */
3213FNIEMOP_DEF(iemOp_punpckldq_Vx_Wx)
3214{
3215 IEMOP_MNEMONIC(punpckldq_Vx_Wx, "punpckldq Vx, Wx");
3216 return FNIEMOP_CALL_1(iemOpCommonSse_LowLow_To_Full, &g_iemAImpl_punpckldq);
3217}
3218
3219/* Opcode 0xf3 0x0f 0x62 - invalid */
3220
3221
3222
3223/** Opcode 0x0f 0x63 - packsswb Pq, Qq */
3224FNIEMOP_STUB(iemOp_packsswb_Pq_Qq);
3225/** Opcode 0x66 0x0f 0x63 - packsswb Vx, Wx */
3226FNIEMOP_STUB(iemOp_packsswb_Vx_Wx);
3227/* Opcode 0xf3 0x0f 0x63 - invalid */
3228
3229/** Opcode 0x0f 0x64 - pcmpgtb Pq, Qq */
3230FNIEMOP_STUB(iemOp_pcmpgtb_Pq_Qq);
3231/** Opcode 0x66 0x0f 0x64 - pcmpgtb Vx, Wx */
3232FNIEMOP_STUB(iemOp_pcmpgtb_Vx_Wx);
3233/* Opcode 0xf3 0x0f 0x64 - invalid */
3234
3235/** Opcode 0x0f 0x65 - pcmpgtw Pq, Qq */
3236FNIEMOP_STUB(iemOp_pcmpgtw_Pq_Qq);
3237/** Opcode 0x66 0x0f 0x65 - pcmpgtw Vx, Wx */
3238FNIEMOP_STUB(iemOp_pcmpgtw_Vx_Wx);
3239/* Opcode 0xf3 0x0f 0x65 - invalid */
3240
3241/** Opcode 0x0f 0x66 - pcmpgtd Pq, Qq */
3242FNIEMOP_STUB(iemOp_pcmpgtd_Pq_Qq);
3243/** Opcode 0x66 0x0f 0x66 - pcmpgtd Vx, Wx */
3244FNIEMOP_STUB(iemOp_pcmpgtd_Vx_Wx);
3245/* Opcode 0xf3 0x0f 0x66 - invalid */
3246
3247/** Opcode 0x0f 0x67 - packuswb Pq, Qq */
3248FNIEMOP_STUB(iemOp_packuswb_Pq_Qq);
3249/** Opcode 0x66 0x0f 0x67 - packuswb Vx, W */
3250FNIEMOP_STUB(iemOp_packuswb_Vx_W);
3251/* Opcode 0xf3 0x0f 0x67 - invalid */
3252
3253
3254/**
3255 * Common worker for MMX instructions on the form:
3256 * pxxxx mm1, mm2/mem64
3257 *
3258 * The 2nd operand is the second half of a register, which in the memory case
3259 * means a 64-bit memory access for MMX, and for SSE a 128-bit aligned access
3260 * where it may read the full 128 bits or only the upper 64 bits.
3261 *
3262 * Exceptions type 4.
3263 */
3264FNIEMOP_DEF_1(iemOpCommonMmx_HighHigh_To_Full, PCIEMOPMEDIAF1H1, pImpl)
3265{
3266 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3267 AssertReturn(pImpl->pfnU64, IEMOP_RAISE_INVALID_OPCODE());
3268 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3269 {
3270 /*
3271 * Register, register.
3272 */
3273 /** @todo testcase: REX.B / REX.R and MMX register indexing. Ignored? */
3274 /** @todo testcase: REX.B / REX.R and segment register indexing. Ignored? */
3275 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3276 IEM_MC_BEGIN(2, 0);
3277 IEM_MC_ARG(uint64_t *, pDst, 0);
3278 IEM_MC_ARG(uint64_t const *, pSrc, 1);
3279 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3280 IEM_MC_PREPARE_FPU_USAGE();
3281 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3282 IEM_MC_REF_MREG_U64_CONST(pSrc, bRm & X86_MODRM_RM_MASK);
3283 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
3284 IEM_MC_ADVANCE_RIP();
3285 IEM_MC_END();
3286 }
3287 else
3288 {
3289 /*
3290 * Register, memory.
3291 */
3292 IEM_MC_BEGIN(2, 2);
3293 IEM_MC_ARG(uint64_t *, pDst, 0);
3294 IEM_MC_LOCAL(uint64_t, uSrc);
3295 IEM_MC_ARG_LOCAL_REF(uint64_t const *, pSrc, uSrc, 1);
3296 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3297
3298 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3299 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3300 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3301 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3302
3303 IEM_MC_PREPARE_FPU_USAGE();
3304 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3305 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
3306
3307 IEM_MC_ADVANCE_RIP();
3308 IEM_MC_END();
3309 }
3310 return VINF_SUCCESS;
3311}
3312
3313
3314/**
3315 * Common worker for SSE2 instructions on the form:
3316 * pxxxx xmm1, xmm2/mem128
3317 *
3318 * The 2nd operand is the second half of a register, which in the memory case
3319 * means a 64-bit memory access for MMX, and for SSE a 128-bit aligned access
3320 * where it may read the full 128 bits or only the upper 64 bits.
3321 *
3322 * Exceptions type 4.
3323 */
3324FNIEMOP_DEF_1(iemOpCommonSse_HighHigh_To_Full, PCIEMOPMEDIAF1H1, pImpl)
3325{
3326 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3327 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3328 {
3329 /*
3330 * Register, register.
3331 */
3332 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3333 IEM_MC_BEGIN(2, 0);
3334 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3335 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
3336 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3337 IEM_MC_PREPARE_SSE_USAGE();
3338 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3339 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3340 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
3341 IEM_MC_ADVANCE_RIP();
3342 IEM_MC_END();
3343 }
3344 else
3345 {
3346 /*
3347 * Register, memory.
3348 */
3349 IEM_MC_BEGIN(2, 2);
3350 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3351 IEM_MC_LOCAL(RTUINT128U, uSrc);
3352 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, pSrc, uSrc, 1);
3353 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3354
3355 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3356 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3357 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3358 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc); /* Most CPUs probably only right high qword */
3359
3360 IEM_MC_PREPARE_SSE_USAGE();
3361 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3362 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
3363
3364 IEM_MC_ADVANCE_RIP();
3365 IEM_MC_END();
3366 }
3367 return VINF_SUCCESS;
3368}
3369
3370
3371/** Opcode 0x0f 0x68 - punpckhbw Pq, Qd */
3372FNIEMOP_DEF(iemOp_punpckhbw_Pq_Qd)
3373{
3374 IEMOP_MNEMONIC(punpckhbw, "punpckhbw Pq, Qd");
3375 return FNIEMOP_CALL_1(iemOpCommonMmx_HighHigh_To_Full, &g_iemAImpl_punpckhbw);
3376}
3377
3378/** Opcode 0x66 0x0f 0x68 - punpckhbw Vx, Wx */
3379FNIEMOP_DEF(iemOp_punpckhbw_Vx_Wx)
3380{
3381 IEMOP_MNEMONIC(vpunpckhbw_Vx_Wx, "vpunpckhbw Vx, Wx");
3382 return FNIEMOP_CALL_1(iemOpCommonSse_HighHigh_To_Full, &g_iemAImpl_punpckhbw);
3383}
3384/* Opcode 0xf3 0x0f 0x68 - invalid */
3385
3386
3387/** Opcode 0x0f 0x69 - punpckhwd Pq, Qd */
3388FNIEMOP_DEF(iemOp_punpckhwd_Pq_Qd)
3389{
3390 IEMOP_MNEMONIC(punpckhwd, "punpckhwd Pq, Qd");
3391 return FNIEMOP_CALL_1(iemOpCommonMmx_HighHigh_To_Full, &g_iemAImpl_punpckhwd);
3392}
3393
3394/** Opcode 0x66 0x0f 0x69 - punpckhwd Vx, Hx, Wx */
3395FNIEMOP_DEF(iemOp_punpckhwd_Vx_Wx)
3396{
3397 IEMOP_MNEMONIC(punpckhwd_Vx_Wx, "punpckhwd Vx, Wx");
3398 return FNIEMOP_CALL_1(iemOpCommonSse_HighHigh_To_Full, &g_iemAImpl_punpckhwd);
3399
3400}
3401/* Opcode 0xf3 0x0f 0x69 - invalid */
3402
3403
3404/** Opcode 0x0f 0x6a - punpckhdq Pq, Qd */
3405FNIEMOP_DEF(iemOp_punpckhdq_Pq_Qd)
3406{
3407 IEMOP_MNEMONIC(punpckhdq, "punpckhdq Pq, Qd");
3408 return FNIEMOP_CALL_1(iemOpCommonMmx_HighHigh_To_Full, &g_iemAImpl_punpckhdq);
3409}
3410
3411/** Opcode 0x66 0x0f 0x6a - punpckhdq Vx, W */
3412FNIEMOP_DEF(iemOp_punpckhdq_Vx_W)
3413{
3414 IEMOP_MNEMONIC(punpckhdq_Vx_W, "punpckhdq Vx, W");
3415 return FNIEMOP_CALL_1(iemOpCommonSse_HighHigh_To_Full, &g_iemAImpl_punpckhdq);
3416}
3417/* Opcode 0xf3 0x0f 0x6a - invalid */
3418
3419
3420/** Opcode 0x0f 0x6b - packssdw Pq, Qd */
3421FNIEMOP_STUB(iemOp_packssdw_Pq_Qd);
3422/** Opcode 0x66 0x0f 0x6b - packssdw Vx, Wx */
3423FNIEMOP_STUB(iemOp_packssdw_Vx_Wx);
3424/* Opcode 0xf3 0x0f 0x6b - invalid */
3425
3426
3427/* Opcode 0x0f 0x6c - invalid */
3428
3429/** Opcode 0x66 0x0f 0x6c - punpcklqdq Vx, Wx */
3430FNIEMOP_DEF(iemOp_punpcklqdq_Vx_Wx)
3431{
3432 IEMOP_MNEMONIC(punpcklqdq, "punpcklqdq Vx, Wx");
3433 return FNIEMOP_CALL_1(iemOpCommonSse_LowLow_To_Full, &g_iemAImpl_punpcklqdq);
3434}
3435
3436/* Opcode 0xf3 0x0f 0x6c - invalid */
3437/* Opcode 0xf2 0x0f 0x6c - invalid */
3438
3439
3440/* Opcode 0x0f 0x6d - invalid */
3441
3442/** Opcode 0x66 0x0f 0x6d - punpckhqdq Vx, W */
3443FNIEMOP_DEF(iemOp_punpckhqdq_Vx_W)
3444{
3445 IEMOP_MNEMONIC(punpckhqdq_Vx_W, "punpckhqdq Vx,W");
3446 return FNIEMOP_CALL_1(iemOpCommonSse_HighHigh_To_Full, &g_iemAImpl_punpckhqdq);
3447}
3448
3449/* Opcode 0xf3 0x0f 0x6d - invalid */
3450
3451
3452FNIEMOP_DEF(iemOp_movd_q_Pd_Ey)
3453{
3454 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3455 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_REX_W)
3456 {
3457 /**
3458 * @opcode 0x6e
3459 * @opcodesub rex.w=1
3460 * @oppfx none
3461 * @opcpuid mmx
3462 * @opgroup og_mmx_datamove
3463 * @opxcpttype 5
3464 * @optest 64-bit / op1=1 op2=2 -> op1=2 ftw=0xff
3465 * @optest 64-bit / op1=0 op2=-42 -> op1=-42 ftw=0xff
3466 */
3467 IEMOP_MNEMONIC2(RM, MOVQ, movq, Pq_WO, Eq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
3468 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3469 {
3470 /* MMX, greg64 */
3471 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3472 IEM_MC_BEGIN(0, 1);
3473 IEM_MC_LOCAL(uint64_t, u64Tmp);
3474
3475 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3476 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3477
3478 IEM_MC_FETCH_GREG_U64(u64Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3479 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u64Tmp);
3480 IEM_MC_FPU_TO_MMX_MODE();
3481
3482 IEM_MC_ADVANCE_RIP();
3483 IEM_MC_END();
3484 }
3485 else
3486 {
3487 /* MMX, [mem64] */
3488 IEM_MC_BEGIN(0, 2);
3489 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3490 IEM_MC_LOCAL(uint64_t, u64Tmp);
3491
3492 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3493 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3494 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3495 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3496
3497 IEM_MC_FETCH_MEM_U64(u64Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3498 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u64Tmp);
3499 IEM_MC_FPU_TO_MMX_MODE();
3500
3501 IEM_MC_ADVANCE_RIP();
3502 IEM_MC_END();
3503 }
3504 }
3505 else
3506 {
3507 /**
3508 * @opdone
3509 * @opcode 0x6e
3510 * @opcodesub rex.w=0
3511 * @oppfx none
3512 * @opcpuid mmx
3513 * @opgroup og_mmx_datamove
3514 * @opxcpttype 5
3515 * @opfunction iemOp_movd_q_Pd_Ey
3516 * @optest op1=1 op2=2 -> op1=2 ftw=0xff
3517 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
3518 */
3519 IEMOP_MNEMONIC2(RM, MOVD, movd, PdZx_WO, Ed, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
3520 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3521 {
3522 /* MMX, greg */
3523 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3524 IEM_MC_BEGIN(0, 1);
3525 IEM_MC_LOCAL(uint64_t, u64Tmp);
3526
3527 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3528 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3529
3530 IEM_MC_FETCH_GREG_U32_ZX_U64(u64Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3531 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u64Tmp);
3532 IEM_MC_FPU_TO_MMX_MODE();
3533
3534 IEM_MC_ADVANCE_RIP();
3535 IEM_MC_END();
3536 }
3537 else
3538 {
3539 /* MMX, [mem] */
3540 IEM_MC_BEGIN(0, 2);
3541 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3542 IEM_MC_LOCAL(uint32_t, u32Tmp);
3543
3544 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3545 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3546 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3547 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3548
3549 IEM_MC_FETCH_MEM_U32(u32Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3550 IEM_MC_STORE_MREG_U32_ZX_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u32Tmp);
3551 IEM_MC_FPU_TO_MMX_MODE();
3552
3553 IEM_MC_ADVANCE_RIP();
3554 IEM_MC_END();
3555 }
3556 }
3557 return VINF_SUCCESS;
3558}
3559
3560FNIEMOP_DEF(iemOp_movd_q_Vy_Ey)
3561{
3562 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3563 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_REX_W)
3564 {
3565 /**
3566 * @opcode 0x6e
3567 * @opcodesub rex.w=1
3568 * @oppfx 0x66
3569 * @opcpuid sse2
3570 * @opgroup og_sse2_simdint_datamove
3571 * @opxcpttype 5
3572 * @optest 64-bit / op1=1 op2=2 -> op1=2
3573 * @optest 64-bit / op1=0 op2=-42 -> op1=-42
3574 */
3575 IEMOP_MNEMONIC2(RM, MOVQ, movq, VqZx_WO, Eq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
3576 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3577 {
3578 /* XMM, greg64 */
3579 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3580 IEM_MC_BEGIN(0, 1);
3581 IEM_MC_LOCAL(uint64_t, u64Tmp);
3582
3583 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3584 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3585
3586 IEM_MC_FETCH_GREG_U64(u64Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3587 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Tmp);
3588
3589 IEM_MC_ADVANCE_RIP();
3590 IEM_MC_END();
3591 }
3592 else
3593 {
3594 /* XMM, [mem64] */
3595 IEM_MC_BEGIN(0, 2);
3596 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3597 IEM_MC_LOCAL(uint64_t, u64Tmp);
3598
3599 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3600 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3601 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3602 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3603
3604 IEM_MC_FETCH_MEM_U64(u64Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3605 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Tmp);
3606
3607 IEM_MC_ADVANCE_RIP();
3608 IEM_MC_END();
3609 }
3610 }
3611 else
3612 {
3613 /**
3614 * @opdone
3615 * @opcode 0x6e
3616 * @opcodesub rex.w=0
3617 * @oppfx 0x66
3618 * @opcpuid sse2
3619 * @opgroup og_sse2_simdint_datamove
3620 * @opxcpttype 5
3621 * @opfunction iemOp_movd_q_Vy_Ey
3622 * @optest op1=1 op2=2 -> op1=2
3623 * @optest op1=0 op2=-42 -> op1=-42
3624 */
3625 IEMOP_MNEMONIC2(RM, MOVD, movd, VdZx_WO, Ed, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
3626 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3627 {
3628 /* XMM, greg32 */
3629 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3630 IEM_MC_BEGIN(0, 1);
3631 IEM_MC_LOCAL(uint32_t, u32Tmp);
3632
3633 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3634 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3635
3636 IEM_MC_FETCH_GREG_U32(u32Tmp, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3637 IEM_MC_STORE_XREG_U32_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Tmp);
3638
3639 IEM_MC_ADVANCE_RIP();
3640 IEM_MC_END();
3641 }
3642 else
3643 {
3644 /* XMM, [mem32] */
3645 IEM_MC_BEGIN(0, 2);
3646 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3647 IEM_MC_LOCAL(uint32_t, u32Tmp);
3648
3649 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3650 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3651 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3652 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3653
3654 IEM_MC_FETCH_MEM_U32(u32Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3655 IEM_MC_STORE_XREG_U32_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Tmp);
3656
3657 IEM_MC_ADVANCE_RIP();
3658 IEM_MC_END();
3659 }
3660 }
3661 return VINF_SUCCESS;
3662}
3663
3664/* Opcode 0xf3 0x0f 0x6e - invalid */
3665
3666
3667/**
3668 * @opcode 0x6f
3669 * @oppfx none
3670 * @opcpuid mmx
3671 * @opgroup og_mmx_datamove
3672 * @opxcpttype 5
3673 * @optest op1=1 op2=2 -> op1=2 ftw=0xff
3674 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
3675 */
3676FNIEMOP_DEF(iemOp_movq_Pq_Qq)
3677{
3678 IEMOP_MNEMONIC2(RM, MOVD, movd, Pq_WO, Qq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
3679 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3680 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3681 {
3682 /*
3683 * Register, register.
3684 */
3685 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3686 IEM_MC_BEGIN(0, 1);
3687 IEM_MC_LOCAL(uint64_t, u64Tmp);
3688
3689 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3690 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3691
3692 IEM_MC_FETCH_MREG_U64(u64Tmp, bRm & X86_MODRM_RM_MASK);
3693 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u64Tmp);
3694 IEM_MC_FPU_TO_MMX_MODE();
3695
3696 IEM_MC_ADVANCE_RIP();
3697 IEM_MC_END();
3698 }
3699 else
3700 {
3701 /*
3702 * Register, memory.
3703 */
3704 IEM_MC_BEGIN(0, 2);
3705 IEM_MC_LOCAL(uint64_t, u64Tmp);
3706 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3707
3708 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3709 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3710 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
3711 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
3712
3713 IEM_MC_FETCH_MEM_U64(u64Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3714 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, u64Tmp);
3715 IEM_MC_FPU_TO_MMX_MODE();
3716
3717 IEM_MC_ADVANCE_RIP();
3718 IEM_MC_END();
3719 }
3720 return VINF_SUCCESS;
3721}
3722
3723/**
3724 * @opcode 0x6f
3725 * @oppfx 0x66
3726 * @opcpuid sse2
3727 * @opgroup og_sse2_simdint_datamove
3728 * @opxcpttype 1
3729 * @optest op1=1 op2=2 -> op1=2
3730 * @optest op1=0 op2=-42 -> op1=-42
3731 */
3732FNIEMOP_DEF(iemOp_movdqa_Vdq_Wdq)
3733{
3734 IEMOP_MNEMONIC2(RM, MOVDQA, movdqa, Vdq_WO, Wdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
3735 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3736 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3737 {
3738 /*
3739 * Register, register.
3740 */
3741 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3742 IEM_MC_BEGIN(0, 0);
3743
3744 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3745 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3746
3747 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
3748 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3749 IEM_MC_ADVANCE_RIP();
3750 IEM_MC_END();
3751 }
3752 else
3753 {
3754 /*
3755 * Register, memory.
3756 */
3757 IEM_MC_BEGIN(0, 2);
3758 IEM_MC_LOCAL(RTUINT128U, u128Tmp);
3759 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3760
3761 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3762 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3763 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3764 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3765
3766 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(u128Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3767 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u128Tmp);
3768
3769 IEM_MC_ADVANCE_RIP();
3770 IEM_MC_END();
3771 }
3772 return VINF_SUCCESS;
3773}
3774
3775/**
3776 * @opcode 0x6f
3777 * @oppfx 0xf3
3778 * @opcpuid sse2
3779 * @opgroup og_sse2_simdint_datamove
3780 * @opxcpttype 4UA
3781 * @optest op1=1 op2=2 -> op1=2
3782 * @optest op1=0 op2=-42 -> op1=-42
3783 */
3784FNIEMOP_DEF(iemOp_movdqu_Vdq_Wdq)
3785{
3786 IEMOP_MNEMONIC2(RM, MOVDQU, movdqu, Vdq_WO, Wdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
3787 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3788 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3789 {
3790 /*
3791 * Register, register.
3792 */
3793 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3794 IEM_MC_BEGIN(0, 0);
3795 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3796 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3797 IEM_MC_COPY_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg,
3798 (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3799 IEM_MC_ADVANCE_RIP();
3800 IEM_MC_END();
3801 }
3802 else
3803 {
3804 /*
3805 * Register, memory.
3806 */
3807 IEM_MC_BEGIN(0, 2);
3808 IEM_MC_LOCAL(RTUINT128U, u128Tmp);
3809 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3810
3811 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3812 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3813 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3814 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
3815 IEM_MC_FETCH_MEM_U128(u128Tmp, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3816 IEM_MC_STORE_XREG_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u128Tmp);
3817
3818 IEM_MC_ADVANCE_RIP();
3819 IEM_MC_END();
3820 }
3821 return VINF_SUCCESS;
3822}
3823
3824
3825/** Opcode 0x0f 0x70 - pshufw Pq, Qq, Ib */
3826FNIEMOP_DEF(iemOp_pshufw_Pq_Qq_Ib)
3827{
3828 IEMOP_MNEMONIC(pshufw_Pq_Qq, "pshufw Pq,Qq,Ib");
3829 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3830 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3831 {
3832 /*
3833 * Register, register.
3834 */
3835 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3836 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3837
3838 IEM_MC_BEGIN(3, 0);
3839 IEM_MC_ARG(uint64_t *, pDst, 0);
3840 IEM_MC_ARG(uint64_t const *, pSrc, 1);
3841 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3842 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT_CHECK_SSE_OR_MMXEXT();
3843 IEM_MC_PREPARE_FPU_USAGE();
3844 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3845 IEM_MC_REF_MREG_U64_CONST(pSrc, bRm & X86_MODRM_RM_MASK);
3846 IEM_MC_CALL_MMX_AIMPL_3(iemAImpl_pshufw, pDst, pSrc, bEvilArg);
3847 IEM_MC_ADVANCE_RIP();
3848 IEM_MC_END();
3849 }
3850 else
3851 {
3852 /*
3853 * Register, memory.
3854 */
3855 IEM_MC_BEGIN(3, 2);
3856 IEM_MC_ARG(uint64_t *, pDst, 0);
3857 IEM_MC_LOCAL(uint64_t, uSrc);
3858 IEM_MC_ARG_LOCAL_REF(uint64_t const *, pSrc, uSrc, 1);
3859 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3860
3861 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3862 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3863 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3864 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3865 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT_CHECK_SSE_OR_MMXEXT();
3866
3867 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3868 IEM_MC_PREPARE_FPU_USAGE();
3869 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
3870 IEM_MC_CALL_MMX_AIMPL_3(iemAImpl_pshufw, pDst, pSrc, bEvilArg);
3871
3872 IEM_MC_ADVANCE_RIP();
3873 IEM_MC_END();
3874 }
3875 return VINF_SUCCESS;
3876}
3877
3878/** Opcode 0x66 0x0f 0x70 - pshufd Vx, Wx, Ib */
3879FNIEMOP_DEF(iemOp_pshufd_Vx_Wx_Ib)
3880{
3881 IEMOP_MNEMONIC(pshufd_Vx_Wx_Ib, "pshufd Vx,Wx,Ib");
3882 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3883 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3884 {
3885 /*
3886 * Register, register.
3887 */
3888 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3889 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3890
3891 IEM_MC_BEGIN(3, 0);
3892 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3893 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
3894 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3895 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3896 IEM_MC_PREPARE_SSE_USAGE();
3897 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3898 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3899 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshufd, pDst, pSrc, bEvilArg);
3900 IEM_MC_ADVANCE_RIP();
3901 IEM_MC_END();
3902 }
3903 else
3904 {
3905 /*
3906 * Register, memory.
3907 */
3908 IEM_MC_BEGIN(3, 2);
3909 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3910 IEM_MC_LOCAL(RTUINT128U, uSrc);
3911 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, pSrc, uSrc, 1);
3912 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3913
3914 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3915 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3916 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3917 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3918 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3919
3920 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3921 IEM_MC_PREPARE_SSE_USAGE();
3922 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3923 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshufd, pDst, pSrc, bEvilArg);
3924
3925 IEM_MC_ADVANCE_RIP();
3926 IEM_MC_END();
3927 }
3928 return VINF_SUCCESS;
3929}
3930
3931/** Opcode 0xf3 0x0f 0x70 - pshufhw Vx, Wx, Ib */
3932FNIEMOP_DEF(iemOp_pshufhw_Vx_Wx_Ib)
3933{
3934 IEMOP_MNEMONIC(pshufhw_Vx_Wx_Ib, "pshufhw Vx,Wx,Ib");
3935 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3936 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3937 {
3938 /*
3939 * Register, register.
3940 */
3941 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3942 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3943
3944 IEM_MC_BEGIN(3, 0);
3945 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3946 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
3947 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3948 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3949 IEM_MC_PREPARE_SSE_USAGE();
3950 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3951 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
3952 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshufhw, pDst, pSrc, bEvilArg);
3953 IEM_MC_ADVANCE_RIP();
3954 IEM_MC_END();
3955 }
3956 else
3957 {
3958 /*
3959 * Register, memory.
3960 */
3961 IEM_MC_BEGIN(3, 2);
3962 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3963 IEM_MC_LOCAL(RTUINT128U, uSrc);
3964 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, pSrc, uSrc, 1);
3965 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
3966
3967 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
3968 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3969 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
3970 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3971 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
3972
3973 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
3974 IEM_MC_PREPARE_SSE_USAGE();
3975 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
3976 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshufhw, pDst, pSrc, bEvilArg);
3977
3978 IEM_MC_ADVANCE_RIP();
3979 IEM_MC_END();
3980 }
3981 return VINF_SUCCESS;
3982}
3983
3984/** Opcode 0xf2 0x0f 0x70 - pshuflw Vx, Wx, Ib */
3985FNIEMOP_DEF(iemOp_pshuflw_Vx_Wx_Ib)
3986{
3987 IEMOP_MNEMONIC(pshuflw_Vx_Wx_Ib, "pshuflw Vx,Wx,Ib");
3988 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
3989 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
3990 {
3991 /*
3992 * Register, register.
3993 */
3994 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
3995 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
3996
3997 IEM_MC_BEGIN(3, 0);
3998 IEM_MC_ARG(PRTUINT128U, pDst, 0);
3999 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
4000 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
4001 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4002 IEM_MC_PREPARE_SSE_USAGE();
4003 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4004 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
4005 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshuflw, pDst, pSrc, bEvilArg);
4006 IEM_MC_ADVANCE_RIP();
4007 IEM_MC_END();
4008 }
4009 else
4010 {
4011 /*
4012 * Register, memory.
4013 */
4014 IEM_MC_BEGIN(3, 2);
4015 IEM_MC_ARG(PRTUINT128U, pDst, 0);
4016 IEM_MC_LOCAL(RTUINT128U, uSrc);
4017 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, pSrc, uSrc, 1);
4018 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4019
4020 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4021 uint8_t bEvil; IEM_OPCODE_GET_NEXT_U8(&bEvil);
4022 IEM_MC_ARG_CONST(uint8_t, bEvilArg, /*=*/ bEvil, 2);
4023 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4024 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4025
4026 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
4027 IEM_MC_PREPARE_SSE_USAGE();
4028 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4029 IEM_MC_CALL_SSE_AIMPL_3(iemAImpl_pshuflw, pDst, pSrc, bEvilArg);
4030
4031 IEM_MC_ADVANCE_RIP();
4032 IEM_MC_END();
4033 }
4034 return VINF_SUCCESS;
4035}
4036
4037
4038/** Opcode 0x0f 0x71 11/2. */
4039FNIEMOP_STUB_1(iemOp_Grp12_psrlw_Nq_Ib, uint8_t, bRm);
4040
4041/** Opcode 0x66 0x0f 0x71 11/2. */
4042FNIEMOP_STUB_1(iemOp_Grp12_psrlw_Ux_Ib, uint8_t, bRm);
4043
4044/** Opcode 0x0f 0x71 11/4. */
4045FNIEMOP_STUB_1(iemOp_Grp12_psraw_Nq_Ib, uint8_t, bRm);
4046
4047/** Opcode 0x66 0x0f 0x71 11/4. */
4048FNIEMOP_STUB_1(iemOp_Grp12_psraw_Ux_Ib, uint8_t, bRm);
4049
4050/** Opcode 0x0f 0x71 11/6. */
4051FNIEMOP_STUB_1(iemOp_Grp12_psllw_Nq_Ib, uint8_t, bRm);
4052
4053/** Opcode 0x66 0x0f 0x71 11/6. */
4054FNIEMOP_STUB_1(iemOp_Grp12_psllw_Ux_Ib, uint8_t, bRm);
4055
4056
4057/**
4058 * Group 12 jump table for register variant.
4059 */
4060IEM_STATIC const PFNIEMOPRM g_apfnGroup12RegReg[] =
4061{
4062 /* /0 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4063 /* /1 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4064 /* /2 */ iemOp_Grp12_psrlw_Nq_Ib, iemOp_Grp12_psrlw_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4065 /* /3 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4066 /* /4 */ iemOp_Grp12_psraw_Nq_Ib, iemOp_Grp12_psraw_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4067 /* /5 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4068 /* /6 */ iemOp_Grp12_psllw_Nq_Ib, iemOp_Grp12_psllw_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4069 /* /7 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8)
4070};
4071AssertCompile(RT_ELEMENTS(g_apfnGroup12RegReg) == 8*4);
4072
4073
4074/** Opcode 0x0f 0x71. */
4075FNIEMOP_DEF(iemOp_Grp12)
4076{
4077 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4078 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4079 /* register, register */
4080 return FNIEMOP_CALL_1(g_apfnGroup12RegReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
4081 + pVCpu->iem.s.idxPrefix], bRm);
4082 return FNIEMOP_CALL_1(iemOp_InvalidWithRMNeedImm8, bRm);
4083}
4084
4085
4086/** Opcode 0x0f 0x72 11/2. */
4087FNIEMOP_STUB_1(iemOp_Grp13_psrld_Nq_Ib, uint8_t, bRm);
4088
4089/** Opcode 0x66 0x0f 0x72 11/2. */
4090FNIEMOP_STUB_1(iemOp_Grp13_psrld_Ux_Ib, uint8_t, bRm);
4091
4092/** Opcode 0x0f 0x72 11/4. */
4093FNIEMOP_STUB_1(iemOp_Grp13_psrad_Nq_Ib, uint8_t, bRm);
4094
4095/** Opcode 0x66 0x0f 0x72 11/4. */
4096FNIEMOP_STUB_1(iemOp_Grp13_psrad_Ux_Ib, uint8_t, bRm);
4097
4098/** Opcode 0x0f 0x72 11/6. */
4099FNIEMOP_STUB_1(iemOp_Grp13_pslld_Nq_Ib, uint8_t, bRm);
4100
4101/** Opcode 0x66 0x0f 0x72 11/6. */
4102FNIEMOP_STUB_1(iemOp_Grp13_pslld_Ux_Ib, uint8_t, bRm);
4103
4104
4105/**
4106 * Group 13 jump table for register variant.
4107 */
4108IEM_STATIC const PFNIEMOPRM g_apfnGroup13RegReg[] =
4109{
4110 /* /0 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4111 /* /1 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4112 /* /2 */ iemOp_Grp13_psrld_Nq_Ib, iemOp_Grp13_psrld_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4113 /* /3 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4114 /* /4 */ iemOp_Grp13_psrad_Nq_Ib, iemOp_Grp13_psrad_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4115 /* /5 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4116 /* /6 */ iemOp_Grp13_pslld_Nq_Ib, iemOp_Grp13_pslld_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4117 /* /7 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8)
4118};
4119AssertCompile(RT_ELEMENTS(g_apfnGroup13RegReg) == 8*4);
4120
4121/** Opcode 0x0f 0x72. */
4122FNIEMOP_DEF(iemOp_Grp13)
4123{
4124 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4125 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4126 /* register, register */
4127 return FNIEMOP_CALL_1(g_apfnGroup13RegReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
4128 + pVCpu->iem.s.idxPrefix], bRm);
4129 return FNIEMOP_CALL_1(iemOp_InvalidWithRMNeedImm8, bRm);
4130}
4131
4132
4133/** Opcode 0x0f 0x73 11/2. */
4134FNIEMOP_STUB_1(iemOp_Grp14_psrlq_Nq_Ib, uint8_t, bRm);
4135
4136/** Opcode 0x66 0x0f 0x73 11/2. */
4137FNIEMOP_STUB_1(iemOp_Grp14_psrlq_Ux_Ib, uint8_t, bRm);
4138
4139/** Opcode 0x66 0x0f 0x73 11/3. */
4140FNIEMOP_STUB_1(iemOp_Grp14_psrldq_Ux_Ib, uint8_t, bRm); //NEXT
4141
4142/** Opcode 0x0f 0x73 11/6. */
4143FNIEMOP_STUB_1(iemOp_Grp14_psllq_Nq_Ib, uint8_t, bRm);
4144
4145/** Opcode 0x66 0x0f 0x73 11/6. */
4146FNIEMOP_STUB_1(iemOp_Grp14_psllq_Ux_Ib, uint8_t, bRm);
4147
4148/** Opcode 0x66 0x0f 0x73 11/7. */
4149FNIEMOP_STUB_1(iemOp_Grp14_pslldq_Ux_Ib, uint8_t, bRm); //NEXT
4150
4151/**
4152 * Group 14 jump table for register variant.
4153 */
4154IEM_STATIC const PFNIEMOPRM g_apfnGroup14RegReg[] =
4155{
4156 /* /0 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4157 /* /1 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4158 /* /2 */ iemOp_Grp14_psrlq_Nq_Ib, iemOp_Grp14_psrlq_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4159 /* /3 */ iemOp_InvalidWithRMNeedImm8, iemOp_Grp14_psrldq_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4160 /* /4 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4161 /* /5 */ IEMOP_X4(iemOp_InvalidWithRMNeedImm8),
4162 /* /6 */ iemOp_Grp14_psllq_Nq_Ib, iemOp_Grp14_psllq_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4163 /* /7 */ iemOp_InvalidWithRMNeedImm8, iemOp_Grp14_pslldq_Ux_Ib, iemOp_InvalidWithRMNeedImm8, iemOp_InvalidWithRMNeedImm8,
4164};
4165AssertCompile(RT_ELEMENTS(g_apfnGroup14RegReg) == 8*4);
4166
4167
4168/** Opcode 0x0f 0x73. */
4169FNIEMOP_DEF(iemOp_Grp14)
4170{
4171 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4172 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4173 /* register, register */
4174 return FNIEMOP_CALL_1(g_apfnGroup14RegReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
4175 + pVCpu->iem.s.idxPrefix], bRm);
4176 return FNIEMOP_CALL_1(iemOp_InvalidWithRMNeedImm8, bRm);
4177}
4178
4179
4180/**
4181 * Common worker for MMX instructions on the form:
4182 * pxxx mm1, mm2/mem64
4183 */
4184FNIEMOP_DEF_1(iemOpCommonMmx_FullFull_To_Full, PCIEMOPMEDIAF2, pImpl)
4185{
4186 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4187 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4188 {
4189 /*
4190 * Register, register.
4191 */
4192 /** @todo testcase: REX.B / REX.R and MMX register indexing. Ignored? */
4193 /** @todo testcase: REX.B / REX.R and segment register indexing. Ignored? */
4194 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4195 IEM_MC_BEGIN(2, 0);
4196 IEM_MC_ARG(uint64_t *, pDst, 0);
4197 IEM_MC_ARG(uint64_t const *, pSrc, 1);
4198 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4199 IEM_MC_PREPARE_FPU_USAGE();
4200 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4201 IEM_MC_REF_MREG_U64_CONST(pSrc, bRm & X86_MODRM_RM_MASK);
4202 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
4203 IEM_MC_ADVANCE_RIP();
4204 IEM_MC_END();
4205 }
4206 else
4207 {
4208 /*
4209 * Register, memory.
4210 */
4211 IEM_MC_BEGIN(2, 2);
4212 IEM_MC_ARG(uint64_t *, pDst, 0);
4213 IEM_MC_LOCAL(uint64_t, uSrc);
4214 IEM_MC_ARG_LOCAL_REF(uint64_t const *, pSrc, uSrc, 1);
4215 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4216
4217 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4218 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4219 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4220 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
4221
4222 IEM_MC_PREPARE_FPU_USAGE();
4223 IEM_MC_REF_MREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4224 IEM_MC_CALL_MMX_AIMPL_2(pImpl->pfnU64, pDst, pSrc);
4225
4226 IEM_MC_ADVANCE_RIP();
4227 IEM_MC_END();
4228 }
4229 return VINF_SUCCESS;
4230}
4231
4232
4233/**
4234 * Common worker for SSE2 instructions on the forms:
4235 * pxxx xmm1, xmm2/mem128
4236 *
4237 * Proper alignment of the 128-bit operand is enforced.
4238 * Exceptions type 4. SSE2 cpuid checks.
4239 */
4240FNIEMOP_DEF_1(iemOpCommonSse2_FullFull_To_Full, PCIEMOPMEDIAF2, pImpl)
4241{
4242 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4243 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4244 {
4245 /*
4246 * Register, register.
4247 */
4248 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4249 IEM_MC_BEGIN(2, 0);
4250 IEM_MC_ARG(PRTUINT128U, pDst, 0);
4251 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
4252 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4253 IEM_MC_PREPARE_SSE_USAGE();
4254 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4255 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
4256 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
4257 IEM_MC_ADVANCE_RIP();
4258 IEM_MC_END();
4259 }
4260 else
4261 {
4262 /*
4263 * Register, memory.
4264 */
4265 IEM_MC_BEGIN(2, 2);
4266 IEM_MC_ARG(PRTUINT128U, pDst, 0);
4267 IEM_MC_LOCAL(RTUINT128U, uSrc);
4268 IEM_MC_ARG_LOCAL_REF(PCRTUINT128U, pSrc, uSrc, 1);
4269 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4270
4271 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4272 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4273 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4274 IEM_MC_FETCH_MEM_U128_ALIGN_SSE(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
4275
4276 IEM_MC_PREPARE_SSE_USAGE();
4277 IEM_MC_REF_XREG_U128(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4278 IEM_MC_CALL_SSE_AIMPL_2(pImpl->pfnU128, pDst, pSrc);
4279
4280 IEM_MC_ADVANCE_RIP();
4281 IEM_MC_END();
4282 }
4283 return VINF_SUCCESS;
4284}
4285
4286
4287/** Opcode 0x0f 0x74 - pcmpeqb Pq, Qq */
4288FNIEMOP_DEF(iemOp_pcmpeqb_Pq_Qq)
4289{
4290 IEMOP_MNEMONIC(pcmpeqb, "pcmpeqb");
4291 return FNIEMOP_CALL_1(iemOpCommonMmx_FullFull_To_Full, &g_iemAImpl_pcmpeqb);
4292}
4293
4294/** Opcode 0x66 0x0f 0x74 - pcmpeqb Vx, Wx */
4295FNIEMOP_DEF(iemOp_pcmpeqb_Vx_Wx)
4296{
4297 IEMOP_MNEMONIC(vpcmpeqb_Vx_Wx, "pcmpeqb");
4298 return FNIEMOP_CALL_1(iemOpCommonSse2_FullFull_To_Full, &g_iemAImpl_pcmpeqb);
4299}
4300
4301/* Opcode 0xf3 0x0f 0x74 - invalid */
4302/* Opcode 0xf2 0x0f 0x74 - invalid */
4303
4304
4305/** Opcode 0x0f 0x75 - pcmpeqw Pq, Qq */
4306FNIEMOP_DEF(iemOp_pcmpeqw_Pq_Qq)
4307{
4308 IEMOP_MNEMONIC(pcmpeqw, "pcmpeqw");
4309 return FNIEMOP_CALL_1(iemOpCommonMmx_FullFull_To_Full, &g_iemAImpl_pcmpeqw);
4310}
4311
4312/** Opcode 0x66 0x0f 0x75 - pcmpeqw Vx, Wx */
4313FNIEMOP_DEF(iemOp_pcmpeqw_Vx_Wx)
4314{
4315 IEMOP_MNEMONIC(pcmpeqw_Vx_Wx, "pcmpeqw");
4316 return FNIEMOP_CALL_1(iemOpCommonSse2_FullFull_To_Full, &g_iemAImpl_pcmpeqw);
4317}
4318
4319/* Opcode 0xf3 0x0f 0x75 - invalid */
4320/* Opcode 0xf2 0x0f 0x75 - invalid */
4321
4322
4323/** Opcode 0x0f 0x76 - pcmpeqd Pq, Qq */
4324FNIEMOP_DEF(iemOp_pcmpeqd_Pq_Qq)
4325{
4326 IEMOP_MNEMONIC(pcmpeqd, "pcmpeqd");
4327 return FNIEMOP_CALL_1(iemOpCommonMmx_FullFull_To_Full, &g_iemAImpl_pcmpeqd);
4328}
4329
4330/** Opcode 0x66 0x0f 0x76 - pcmpeqd Vx, Wx */
4331FNIEMOP_DEF(iemOp_pcmpeqd_Vx_Wx)
4332{
4333 IEMOP_MNEMONIC(pcmpeqd_Vx_Wx, "vpcmpeqd");
4334 return FNIEMOP_CALL_1(iemOpCommonSse2_FullFull_To_Full, &g_iemAImpl_pcmpeqd);
4335}
4336
4337/* Opcode 0xf3 0x0f 0x76 - invalid */
4338/* Opcode 0xf2 0x0f 0x76 - invalid */
4339
4340
4341/** Opcode 0x0f 0x77 - emms (vex has vzeroall and vzeroupper here) */
4342FNIEMOP_STUB(iemOp_emms);
4343/* Opcode 0x66 0x0f 0x77 - invalid */
4344/* Opcode 0xf3 0x0f 0x77 - invalid */
4345/* Opcode 0xf2 0x0f 0x77 - invalid */
4346
4347/** Opcode 0x0f 0x78 - VMREAD Ey, Gy */
4348FNIEMOP_STUB(iemOp_vmread_Ey_Gy);
4349/* Opcode 0x66 0x0f 0x78 - AMD Group 17 */
4350FNIEMOP_STUB(iemOp_AmdGrp17);
4351/* Opcode 0xf3 0x0f 0x78 - invalid */
4352/* Opcode 0xf2 0x0f 0x78 - invalid */
4353
4354/** Opcode 0x0f 0x79 - VMWRITE Gy, Ey */
4355FNIEMOP_STUB(iemOp_vmwrite_Gy_Ey);
4356/* Opcode 0x66 0x0f 0x79 - invalid */
4357/* Opcode 0xf3 0x0f 0x79 - invalid */
4358/* Opcode 0xf2 0x0f 0x79 - invalid */
4359
4360/* Opcode 0x0f 0x7a - invalid */
4361/* Opcode 0x66 0x0f 0x7a - invalid */
4362/* Opcode 0xf3 0x0f 0x7a - invalid */
4363/* Opcode 0xf2 0x0f 0x7a - invalid */
4364
4365/* Opcode 0x0f 0x7b - invalid */
4366/* Opcode 0x66 0x0f 0x7b - invalid */
4367/* Opcode 0xf3 0x0f 0x7b - invalid */
4368/* Opcode 0xf2 0x0f 0x7b - invalid */
4369
4370/* Opcode 0x0f 0x7c - invalid */
4371/** Opcode 0x66 0x0f 0x7c - haddpd Vpd, Wpd */
4372FNIEMOP_STUB(iemOp_haddpd_Vpd_Wpd);
4373/* Opcode 0xf3 0x0f 0x7c - invalid */
4374/** Opcode 0xf2 0x0f 0x7c - haddps Vps, Wps */
4375FNIEMOP_STUB(iemOp_haddps_Vps_Wps);
4376
4377/* Opcode 0x0f 0x7d - invalid */
4378/** Opcode 0x66 0x0f 0x7d - hsubpd Vpd, Wpd */
4379FNIEMOP_STUB(iemOp_hsubpd_Vpd_Wpd);
4380/* Opcode 0xf3 0x0f 0x7d - invalid */
4381/** Opcode 0xf2 0x0f 0x7d - hsubps Vps, Wps */
4382FNIEMOP_STUB(iemOp_hsubps_Vps_Wps);
4383
4384
4385/** Opcode 0x0f 0x7e - movd_q Ey, Pd */
4386FNIEMOP_DEF(iemOp_movd_q_Ey_Pd)
4387{
4388 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4389 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_REX_W)
4390 {
4391 /**
4392 * @opcode 0x7e
4393 * @opcodesub rex.w=1
4394 * @oppfx none
4395 * @opcpuid mmx
4396 * @opgroup og_mmx_datamove
4397 * @opxcpttype 5
4398 * @optest 64-bit / op1=1 op2=2 -> op1=2 ftw=0xff
4399 * @optest 64-bit / op1=0 op2=-42 -> op1=-42 ftw=0xff
4400 */
4401 IEMOP_MNEMONIC2(MR, MOVQ, movq, Eq_WO, Pq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
4402 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4403 {
4404 /* greg64, MMX */
4405 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4406 IEM_MC_BEGIN(0, 1);
4407 IEM_MC_LOCAL(uint64_t, u64Tmp);
4408
4409 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4410 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
4411
4412 IEM_MC_FETCH_MREG_U64(u64Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4413 IEM_MC_STORE_GREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u64Tmp);
4414 IEM_MC_FPU_TO_MMX_MODE();
4415
4416 IEM_MC_ADVANCE_RIP();
4417 IEM_MC_END();
4418 }
4419 else
4420 {
4421 /* [mem64], MMX */
4422 IEM_MC_BEGIN(0, 2);
4423 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4424 IEM_MC_LOCAL(uint64_t, u64Tmp);
4425
4426 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4427 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4428 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4429 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
4430
4431 IEM_MC_FETCH_MREG_U64(u64Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4432 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u64Tmp);
4433 IEM_MC_FPU_TO_MMX_MODE();
4434
4435 IEM_MC_ADVANCE_RIP();
4436 IEM_MC_END();
4437 }
4438 }
4439 else
4440 {
4441 /**
4442 * @opdone
4443 * @opcode 0x7e
4444 * @opcodesub rex.w=0
4445 * @oppfx none
4446 * @opcpuid mmx
4447 * @opgroup og_mmx_datamove
4448 * @opxcpttype 5
4449 * @opfunction iemOp_movd_q_Pd_Ey
4450 * @optest op1=1 op2=2 -> op1=2 ftw=0xff
4451 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
4452 */
4453 IEMOP_MNEMONIC2(MR, MOVD, movd, Ed_WO, Pd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
4454 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4455 {
4456 /* greg32, MMX */
4457 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4458 IEM_MC_BEGIN(0, 1);
4459 IEM_MC_LOCAL(uint32_t, u32Tmp);
4460
4461 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4462 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
4463
4464 IEM_MC_FETCH_MREG_U32(u32Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4465 IEM_MC_STORE_GREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u32Tmp);
4466 IEM_MC_FPU_TO_MMX_MODE();
4467
4468 IEM_MC_ADVANCE_RIP();
4469 IEM_MC_END();
4470 }
4471 else
4472 {
4473 /* [mem32], MMX */
4474 IEM_MC_BEGIN(0, 2);
4475 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4476 IEM_MC_LOCAL(uint32_t, u32Tmp);
4477
4478 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4479 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4480 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4481 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
4482
4483 IEM_MC_FETCH_MREG_U32(u32Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4484 IEM_MC_STORE_MEM_U32(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u32Tmp);
4485 IEM_MC_FPU_TO_MMX_MODE();
4486
4487 IEM_MC_ADVANCE_RIP();
4488 IEM_MC_END();
4489 }
4490 }
4491 return VINF_SUCCESS;
4492
4493}
4494
4495
4496FNIEMOP_DEF(iemOp_movd_q_Ey_Vy)
4497{
4498 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4499 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_REX_W)
4500 {
4501 /**
4502 * @opcode 0x7e
4503 * @opcodesub rex.w=1
4504 * @oppfx 0x66
4505 * @opcpuid sse2
4506 * @opgroup og_sse2_simdint_datamove
4507 * @opxcpttype 5
4508 * @optest 64-bit / op1=1 op2=2 -> op1=2
4509 * @optest 64-bit / op1=0 op2=-42 -> op1=-42
4510 */
4511 IEMOP_MNEMONIC2(MR, MOVQ, movq, Eq_WO, Vq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
4512 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4513 {
4514 /* greg64, XMM */
4515 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4516 IEM_MC_BEGIN(0, 1);
4517 IEM_MC_LOCAL(uint64_t, u64Tmp);
4518
4519 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4520 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4521
4522 IEM_MC_FETCH_XREG_U64(u64Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4523 IEM_MC_STORE_GREG_U64((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u64Tmp);
4524
4525 IEM_MC_ADVANCE_RIP();
4526 IEM_MC_END();
4527 }
4528 else
4529 {
4530 /* [mem64], XMM */
4531 IEM_MC_BEGIN(0, 2);
4532 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4533 IEM_MC_LOCAL(uint64_t, u64Tmp);
4534
4535 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4536 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4537 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4538 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4539
4540 IEM_MC_FETCH_XREG_U64(u64Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4541 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u64Tmp);
4542
4543 IEM_MC_ADVANCE_RIP();
4544 IEM_MC_END();
4545 }
4546 }
4547 else
4548 {
4549 /**
4550 * @opdone
4551 * @opcode 0x7e
4552 * @opcodesub rex.w=0
4553 * @oppfx 0x66
4554 * @opcpuid sse2
4555 * @opgroup og_sse2_simdint_datamove
4556 * @opxcpttype 5
4557 * @opfunction iemOp_movd_q_Vy_Ey
4558 * @optest op1=1 op2=2 -> op1=2
4559 * @optest op1=0 op2=-42 -> op1=-42
4560 */
4561 IEMOP_MNEMONIC2(MR, MOVD, movd, Ed_WO, Vd, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OZ_PFX);
4562 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4563 {
4564 /* greg32, XMM */
4565 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4566 IEM_MC_BEGIN(0, 1);
4567 IEM_MC_LOCAL(uint32_t, u32Tmp);
4568
4569 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4570 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4571
4572 IEM_MC_FETCH_XREG_U32(u32Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4573 IEM_MC_STORE_GREG_U32((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, u32Tmp);
4574
4575 IEM_MC_ADVANCE_RIP();
4576 IEM_MC_END();
4577 }
4578 else
4579 {
4580 /* [mem32], XMM */
4581 IEM_MC_BEGIN(0, 2);
4582 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4583 IEM_MC_LOCAL(uint32_t, u32Tmp);
4584
4585 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4586 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4587 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4588 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4589
4590 IEM_MC_FETCH_XREG_U32(u32Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4591 IEM_MC_STORE_MEM_U32(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u32Tmp);
4592
4593 IEM_MC_ADVANCE_RIP();
4594 IEM_MC_END();
4595 }
4596 }
4597 return VINF_SUCCESS;
4598
4599}
4600
4601/**
4602 * @opcode 0x7e
4603 * @oppfx 0xf3
4604 * @opcpuid sse2
4605 * @opgroup og_sse2_pcksclr_datamove
4606 * @opxcpttype none
4607 * @optest op1=1 op2=2 -> op1=2
4608 * @optest op1=0 op2=-42 -> op1=-42
4609 */
4610FNIEMOP_DEF(iemOp_movq_Vq_Wq)
4611{
4612 IEMOP_MNEMONIC2(RM, MOVQ, movq, VqZx_WO, Wq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
4613 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4614 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4615 {
4616 /*
4617 * Register, register.
4618 */
4619 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4620 IEM_MC_BEGIN(0, 2);
4621 IEM_MC_LOCAL(uint64_t, uSrc);
4622
4623 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4624 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
4625
4626 IEM_MC_FETCH_XREG_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
4627 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
4628
4629 IEM_MC_ADVANCE_RIP();
4630 IEM_MC_END();
4631 }
4632 else
4633 {
4634 /*
4635 * Memory, register.
4636 */
4637 IEM_MC_BEGIN(0, 2);
4638 IEM_MC_LOCAL(uint64_t, uSrc);
4639 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4640
4641 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4642 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4643 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4644 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
4645
4646 IEM_MC_FETCH_MEM_U64(uSrc, pVCpu->iem.s.iEffSeg, GCPtrEffSrc);
4647 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
4648
4649 IEM_MC_ADVANCE_RIP();
4650 IEM_MC_END();
4651 }
4652 return VINF_SUCCESS;
4653}
4654
4655/* Opcode 0xf2 0x0f 0x7e - invalid */
4656
4657
4658/** Opcode 0x0f 0x7f - movq Qq, Pq */
4659FNIEMOP_DEF(iemOp_movq_Qq_Pq)
4660{
4661 IEMOP_MNEMONIC(movq_Qq_Pq, "movq Qq,Pq");
4662 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4663 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4664 {
4665 /*
4666 * Register, register.
4667 */
4668 /** @todo testcase: REX.B / REX.R and MMX register indexing. Ignored? */
4669 /** @todo testcase: REX.B / REX.R and segment register indexing. Ignored? */
4670 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4671 IEM_MC_BEGIN(0, 1);
4672 IEM_MC_LOCAL(uint64_t, u64Tmp);
4673 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4674 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
4675 IEM_MC_FETCH_MREG_U64(u64Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4676 IEM_MC_STORE_MREG_U64(bRm & X86_MODRM_RM_MASK, u64Tmp);
4677 IEM_MC_ADVANCE_RIP();
4678 IEM_MC_END();
4679 }
4680 else
4681 {
4682 /*
4683 * Register, memory.
4684 */
4685 IEM_MC_BEGIN(0, 2);
4686 IEM_MC_LOCAL(uint64_t, u64Tmp);
4687 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4688
4689 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4690 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4691 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
4692 IEM_MC_ACTUALIZE_FPU_STATE_FOR_READ();
4693
4694 IEM_MC_FETCH_MREG_U64(u64Tmp, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
4695 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u64Tmp);
4696
4697 IEM_MC_ADVANCE_RIP();
4698 IEM_MC_END();
4699 }
4700 return VINF_SUCCESS;
4701}
4702
4703/** Opcode 0x66 0x0f 0x7f - movdqa Wx,Vx */
4704FNIEMOP_DEF(iemOp_movdqa_Wx_Vx)
4705{
4706 IEMOP_MNEMONIC(movdqa_Wdq_Vdq, "movdqa Wx,Vx");
4707 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4708 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4709 {
4710 /*
4711 * Register, register.
4712 */
4713 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4714 IEM_MC_BEGIN(0, 0);
4715 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4716 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
4717 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
4718 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4719 IEM_MC_ADVANCE_RIP();
4720 IEM_MC_END();
4721 }
4722 else
4723 {
4724 /*
4725 * Register, memory.
4726 */
4727 IEM_MC_BEGIN(0, 2);
4728 IEM_MC_LOCAL(RTUINT128U, u128Tmp);
4729 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4730
4731 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4732 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4733 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4734 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4735
4736 IEM_MC_FETCH_XREG_U128(u128Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4737 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u128Tmp);
4738
4739 IEM_MC_ADVANCE_RIP();
4740 IEM_MC_END();
4741 }
4742 return VINF_SUCCESS;
4743}
4744
4745/** Opcode 0xf3 0x0f 0x7f - movdqu Wx,Vx */
4746FNIEMOP_DEF(iemOp_movdqu_Wx_Vx)
4747{
4748 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
4749 IEMOP_MNEMONIC(movdqu_Wdq_Vdq, "movdqu Wx,Vx");
4750 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
4751 {
4752 /*
4753 * Register, register.
4754 */
4755 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4756 IEM_MC_BEGIN(0, 0);
4757 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4758 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
4759 IEM_MC_COPY_XREG_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB,
4760 ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4761 IEM_MC_ADVANCE_RIP();
4762 IEM_MC_END();
4763 }
4764 else
4765 {
4766 /*
4767 * Register, memory.
4768 */
4769 IEM_MC_BEGIN(0, 2);
4770 IEM_MC_LOCAL(RTUINT128U, u128Tmp);
4771 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
4772
4773 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
4774 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4775 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
4776 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
4777
4778 IEM_MC_FETCH_XREG_U128(u128Tmp, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
4779 IEM_MC_STORE_MEM_U128(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, u128Tmp);
4780
4781 IEM_MC_ADVANCE_RIP();
4782 IEM_MC_END();
4783 }
4784 return VINF_SUCCESS;
4785}
4786
4787/* Opcode 0xf2 0x0f 0x7f - invalid */
4788
4789
4790
4791/** Opcode 0x0f 0x80. */
4792FNIEMOP_DEF(iemOp_jo_Jv)
4793{
4794 IEMOP_MNEMONIC(jo_Jv, "jo Jv");
4795 IEMOP_HLP_MIN_386();
4796 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4797 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4798 {
4799 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4800 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4801
4802 IEM_MC_BEGIN(0, 0);
4803 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
4804 IEM_MC_REL_JMP_S16(i16Imm);
4805 } IEM_MC_ELSE() {
4806 IEM_MC_ADVANCE_RIP();
4807 } IEM_MC_ENDIF();
4808 IEM_MC_END();
4809 }
4810 else
4811 {
4812 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4813 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4814
4815 IEM_MC_BEGIN(0, 0);
4816 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
4817 IEM_MC_REL_JMP_S32(i32Imm);
4818 } IEM_MC_ELSE() {
4819 IEM_MC_ADVANCE_RIP();
4820 } IEM_MC_ENDIF();
4821 IEM_MC_END();
4822 }
4823 return VINF_SUCCESS;
4824}
4825
4826
4827/** Opcode 0x0f 0x81. */
4828FNIEMOP_DEF(iemOp_jno_Jv)
4829{
4830 IEMOP_MNEMONIC(jno_Jv, "jno Jv");
4831 IEMOP_HLP_MIN_386();
4832 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4833 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4834 {
4835 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4836 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4837
4838 IEM_MC_BEGIN(0, 0);
4839 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
4840 IEM_MC_ADVANCE_RIP();
4841 } IEM_MC_ELSE() {
4842 IEM_MC_REL_JMP_S16(i16Imm);
4843 } IEM_MC_ENDIF();
4844 IEM_MC_END();
4845 }
4846 else
4847 {
4848 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4849 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4850
4851 IEM_MC_BEGIN(0, 0);
4852 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
4853 IEM_MC_ADVANCE_RIP();
4854 } IEM_MC_ELSE() {
4855 IEM_MC_REL_JMP_S32(i32Imm);
4856 } IEM_MC_ENDIF();
4857 IEM_MC_END();
4858 }
4859 return VINF_SUCCESS;
4860}
4861
4862
4863/** Opcode 0x0f 0x82. */
4864FNIEMOP_DEF(iemOp_jc_Jv)
4865{
4866 IEMOP_MNEMONIC(jc_Jv, "jc/jb/jnae Jv");
4867 IEMOP_HLP_MIN_386();
4868 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4869 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4870 {
4871 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4872 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4873
4874 IEM_MC_BEGIN(0, 0);
4875 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
4876 IEM_MC_REL_JMP_S16(i16Imm);
4877 } IEM_MC_ELSE() {
4878 IEM_MC_ADVANCE_RIP();
4879 } IEM_MC_ENDIF();
4880 IEM_MC_END();
4881 }
4882 else
4883 {
4884 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4885 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4886
4887 IEM_MC_BEGIN(0, 0);
4888 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
4889 IEM_MC_REL_JMP_S32(i32Imm);
4890 } IEM_MC_ELSE() {
4891 IEM_MC_ADVANCE_RIP();
4892 } IEM_MC_ENDIF();
4893 IEM_MC_END();
4894 }
4895 return VINF_SUCCESS;
4896}
4897
4898
4899/** Opcode 0x0f 0x83. */
4900FNIEMOP_DEF(iemOp_jnc_Jv)
4901{
4902 IEMOP_MNEMONIC(jnc_Jv, "jnc/jnb/jae Jv");
4903 IEMOP_HLP_MIN_386();
4904 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4905 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4906 {
4907 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4908 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4909
4910 IEM_MC_BEGIN(0, 0);
4911 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
4912 IEM_MC_ADVANCE_RIP();
4913 } IEM_MC_ELSE() {
4914 IEM_MC_REL_JMP_S16(i16Imm);
4915 } IEM_MC_ENDIF();
4916 IEM_MC_END();
4917 }
4918 else
4919 {
4920 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4921 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4922
4923 IEM_MC_BEGIN(0, 0);
4924 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
4925 IEM_MC_ADVANCE_RIP();
4926 } IEM_MC_ELSE() {
4927 IEM_MC_REL_JMP_S32(i32Imm);
4928 } IEM_MC_ENDIF();
4929 IEM_MC_END();
4930 }
4931 return VINF_SUCCESS;
4932}
4933
4934
4935/** Opcode 0x0f 0x84. */
4936FNIEMOP_DEF(iemOp_je_Jv)
4937{
4938 IEMOP_MNEMONIC(je_Jv, "je/jz Jv");
4939 IEMOP_HLP_MIN_386();
4940 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4941 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4942 {
4943 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4944 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4945
4946 IEM_MC_BEGIN(0, 0);
4947 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
4948 IEM_MC_REL_JMP_S16(i16Imm);
4949 } IEM_MC_ELSE() {
4950 IEM_MC_ADVANCE_RIP();
4951 } IEM_MC_ENDIF();
4952 IEM_MC_END();
4953 }
4954 else
4955 {
4956 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4957 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4958
4959 IEM_MC_BEGIN(0, 0);
4960 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
4961 IEM_MC_REL_JMP_S32(i32Imm);
4962 } IEM_MC_ELSE() {
4963 IEM_MC_ADVANCE_RIP();
4964 } IEM_MC_ENDIF();
4965 IEM_MC_END();
4966 }
4967 return VINF_SUCCESS;
4968}
4969
4970
4971/** Opcode 0x0f 0x85. */
4972FNIEMOP_DEF(iemOp_jne_Jv)
4973{
4974 IEMOP_MNEMONIC(jne_Jv, "jne/jnz Jv");
4975 IEMOP_HLP_MIN_386();
4976 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
4977 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
4978 {
4979 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
4980 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4981
4982 IEM_MC_BEGIN(0, 0);
4983 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
4984 IEM_MC_ADVANCE_RIP();
4985 } IEM_MC_ELSE() {
4986 IEM_MC_REL_JMP_S16(i16Imm);
4987 } IEM_MC_ENDIF();
4988 IEM_MC_END();
4989 }
4990 else
4991 {
4992 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
4993 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
4994
4995 IEM_MC_BEGIN(0, 0);
4996 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
4997 IEM_MC_ADVANCE_RIP();
4998 } IEM_MC_ELSE() {
4999 IEM_MC_REL_JMP_S32(i32Imm);
5000 } IEM_MC_ENDIF();
5001 IEM_MC_END();
5002 }
5003 return VINF_SUCCESS;
5004}
5005
5006
5007/** Opcode 0x0f 0x86. */
5008FNIEMOP_DEF(iemOp_jbe_Jv)
5009{
5010 IEMOP_MNEMONIC(jbe_Jv, "jbe/jna Jv");
5011 IEMOP_HLP_MIN_386();
5012 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5013 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5014 {
5015 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5016 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5017
5018 IEM_MC_BEGIN(0, 0);
5019 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5020 IEM_MC_REL_JMP_S16(i16Imm);
5021 } IEM_MC_ELSE() {
5022 IEM_MC_ADVANCE_RIP();
5023 } IEM_MC_ENDIF();
5024 IEM_MC_END();
5025 }
5026 else
5027 {
5028 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5029 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5030
5031 IEM_MC_BEGIN(0, 0);
5032 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5033 IEM_MC_REL_JMP_S32(i32Imm);
5034 } IEM_MC_ELSE() {
5035 IEM_MC_ADVANCE_RIP();
5036 } IEM_MC_ENDIF();
5037 IEM_MC_END();
5038 }
5039 return VINF_SUCCESS;
5040}
5041
5042
5043/** Opcode 0x0f 0x87. */
5044FNIEMOP_DEF(iemOp_jnbe_Jv)
5045{
5046 IEMOP_MNEMONIC(ja_Jv, "jnbe/ja Jv");
5047 IEMOP_HLP_MIN_386();
5048 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5049 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5050 {
5051 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5052 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5053
5054 IEM_MC_BEGIN(0, 0);
5055 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5056 IEM_MC_ADVANCE_RIP();
5057 } IEM_MC_ELSE() {
5058 IEM_MC_REL_JMP_S16(i16Imm);
5059 } IEM_MC_ENDIF();
5060 IEM_MC_END();
5061 }
5062 else
5063 {
5064 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5065 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5066
5067 IEM_MC_BEGIN(0, 0);
5068 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5069 IEM_MC_ADVANCE_RIP();
5070 } IEM_MC_ELSE() {
5071 IEM_MC_REL_JMP_S32(i32Imm);
5072 } IEM_MC_ENDIF();
5073 IEM_MC_END();
5074 }
5075 return VINF_SUCCESS;
5076}
5077
5078
5079/** Opcode 0x0f 0x88. */
5080FNIEMOP_DEF(iemOp_js_Jv)
5081{
5082 IEMOP_MNEMONIC(js_Jv, "js Jv");
5083 IEMOP_HLP_MIN_386();
5084 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5085 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5086 {
5087 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5088 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5089
5090 IEM_MC_BEGIN(0, 0);
5091 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5092 IEM_MC_REL_JMP_S16(i16Imm);
5093 } IEM_MC_ELSE() {
5094 IEM_MC_ADVANCE_RIP();
5095 } IEM_MC_ENDIF();
5096 IEM_MC_END();
5097 }
5098 else
5099 {
5100 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5101 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5102
5103 IEM_MC_BEGIN(0, 0);
5104 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5105 IEM_MC_REL_JMP_S32(i32Imm);
5106 } IEM_MC_ELSE() {
5107 IEM_MC_ADVANCE_RIP();
5108 } IEM_MC_ENDIF();
5109 IEM_MC_END();
5110 }
5111 return VINF_SUCCESS;
5112}
5113
5114
5115/** Opcode 0x0f 0x89. */
5116FNIEMOP_DEF(iemOp_jns_Jv)
5117{
5118 IEMOP_MNEMONIC(jns_Jv, "jns Jv");
5119 IEMOP_HLP_MIN_386();
5120 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5121 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5122 {
5123 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5124 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5125
5126 IEM_MC_BEGIN(0, 0);
5127 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5128 IEM_MC_ADVANCE_RIP();
5129 } IEM_MC_ELSE() {
5130 IEM_MC_REL_JMP_S16(i16Imm);
5131 } IEM_MC_ENDIF();
5132 IEM_MC_END();
5133 }
5134 else
5135 {
5136 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5137 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5138
5139 IEM_MC_BEGIN(0, 0);
5140 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5141 IEM_MC_ADVANCE_RIP();
5142 } IEM_MC_ELSE() {
5143 IEM_MC_REL_JMP_S32(i32Imm);
5144 } IEM_MC_ENDIF();
5145 IEM_MC_END();
5146 }
5147 return VINF_SUCCESS;
5148}
5149
5150
5151/** Opcode 0x0f 0x8a. */
5152FNIEMOP_DEF(iemOp_jp_Jv)
5153{
5154 IEMOP_MNEMONIC(jp_Jv, "jp Jv");
5155 IEMOP_HLP_MIN_386();
5156 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5157 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5158 {
5159 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5160 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5161
5162 IEM_MC_BEGIN(0, 0);
5163 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5164 IEM_MC_REL_JMP_S16(i16Imm);
5165 } IEM_MC_ELSE() {
5166 IEM_MC_ADVANCE_RIP();
5167 } IEM_MC_ENDIF();
5168 IEM_MC_END();
5169 }
5170 else
5171 {
5172 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5173 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5174
5175 IEM_MC_BEGIN(0, 0);
5176 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5177 IEM_MC_REL_JMP_S32(i32Imm);
5178 } IEM_MC_ELSE() {
5179 IEM_MC_ADVANCE_RIP();
5180 } IEM_MC_ENDIF();
5181 IEM_MC_END();
5182 }
5183 return VINF_SUCCESS;
5184}
5185
5186
5187/** Opcode 0x0f 0x8b. */
5188FNIEMOP_DEF(iemOp_jnp_Jv)
5189{
5190 IEMOP_MNEMONIC(jnp_Jv, "jnp Jv");
5191 IEMOP_HLP_MIN_386();
5192 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5193 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5194 {
5195 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5196 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5197
5198 IEM_MC_BEGIN(0, 0);
5199 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5200 IEM_MC_ADVANCE_RIP();
5201 } IEM_MC_ELSE() {
5202 IEM_MC_REL_JMP_S16(i16Imm);
5203 } IEM_MC_ENDIF();
5204 IEM_MC_END();
5205 }
5206 else
5207 {
5208 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5209 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5210
5211 IEM_MC_BEGIN(0, 0);
5212 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5213 IEM_MC_ADVANCE_RIP();
5214 } IEM_MC_ELSE() {
5215 IEM_MC_REL_JMP_S32(i32Imm);
5216 } IEM_MC_ENDIF();
5217 IEM_MC_END();
5218 }
5219 return VINF_SUCCESS;
5220}
5221
5222
5223/** Opcode 0x0f 0x8c. */
5224FNIEMOP_DEF(iemOp_jl_Jv)
5225{
5226 IEMOP_MNEMONIC(jl_Jv, "jl/jnge Jv");
5227 IEMOP_HLP_MIN_386();
5228 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5229 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5230 {
5231 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5232 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5233
5234 IEM_MC_BEGIN(0, 0);
5235 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5236 IEM_MC_REL_JMP_S16(i16Imm);
5237 } IEM_MC_ELSE() {
5238 IEM_MC_ADVANCE_RIP();
5239 } IEM_MC_ENDIF();
5240 IEM_MC_END();
5241 }
5242 else
5243 {
5244 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5245 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5246
5247 IEM_MC_BEGIN(0, 0);
5248 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5249 IEM_MC_REL_JMP_S32(i32Imm);
5250 } IEM_MC_ELSE() {
5251 IEM_MC_ADVANCE_RIP();
5252 } IEM_MC_ENDIF();
5253 IEM_MC_END();
5254 }
5255 return VINF_SUCCESS;
5256}
5257
5258
5259/** Opcode 0x0f 0x8d. */
5260FNIEMOP_DEF(iemOp_jnl_Jv)
5261{
5262 IEMOP_MNEMONIC(jge_Jv, "jnl/jge Jv");
5263 IEMOP_HLP_MIN_386();
5264 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5265 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5266 {
5267 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5268 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5269
5270 IEM_MC_BEGIN(0, 0);
5271 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5272 IEM_MC_ADVANCE_RIP();
5273 } IEM_MC_ELSE() {
5274 IEM_MC_REL_JMP_S16(i16Imm);
5275 } IEM_MC_ENDIF();
5276 IEM_MC_END();
5277 }
5278 else
5279 {
5280 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5281 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5282
5283 IEM_MC_BEGIN(0, 0);
5284 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5285 IEM_MC_ADVANCE_RIP();
5286 } IEM_MC_ELSE() {
5287 IEM_MC_REL_JMP_S32(i32Imm);
5288 } IEM_MC_ENDIF();
5289 IEM_MC_END();
5290 }
5291 return VINF_SUCCESS;
5292}
5293
5294
5295/** Opcode 0x0f 0x8e. */
5296FNIEMOP_DEF(iemOp_jle_Jv)
5297{
5298 IEMOP_MNEMONIC(jle_Jv, "jle/jng Jv");
5299 IEMOP_HLP_MIN_386();
5300 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5301 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5302 {
5303 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5304 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5305
5306 IEM_MC_BEGIN(0, 0);
5307 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5308 IEM_MC_REL_JMP_S16(i16Imm);
5309 } IEM_MC_ELSE() {
5310 IEM_MC_ADVANCE_RIP();
5311 } IEM_MC_ENDIF();
5312 IEM_MC_END();
5313 }
5314 else
5315 {
5316 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5317 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5318
5319 IEM_MC_BEGIN(0, 0);
5320 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5321 IEM_MC_REL_JMP_S32(i32Imm);
5322 } IEM_MC_ELSE() {
5323 IEM_MC_ADVANCE_RIP();
5324 } IEM_MC_ENDIF();
5325 IEM_MC_END();
5326 }
5327 return VINF_SUCCESS;
5328}
5329
5330
5331/** Opcode 0x0f 0x8f. */
5332FNIEMOP_DEF(iemOp_jnle_Jv)
5333{
5334 IEMOP_MNEMONIC(jg_Jv, "jnle/jg Jv");
5335 IEMOP_HLP_MIN_386();
5336 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
5337 if (pVCpu->iem.s.enmEffOpSize == IEMMODE_16BIT)
5338 {
5339 int16_t i16Imm; IEM_OPCODE_GET_NEXT_S16(&i16Imm);
5340 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5341
5342 IEM_MC_BEGIN(0, 0);
5343 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5344 IEM_MC_ADVANCE_RIP();
5345 } IEM_MC_ELSE() {
5346 IEM_MC_REL_JMP_S16(i16Imm);
5347 } IEM_MC_ENDIF();
5348 IEM_MC_END();
5349 }
5350 else
5351 {
5352 int32_t i32Imm; IEM_OPCODE_GET_NEXT_S32(&i32Imm);
5353 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5354
5355 IEM_MC_BEGIN(0, 0);
5356 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5357 IEM_MC_ADVANCE_RIP();
5358 } IEM_MC_ELSE() {
5359 IEM_MC_REL_JMP_S32(i32Imm);
5360 } IEM_MC_ENDIF();
5361 IEM_MC_END();
5362 }
5363 return VINF_SUCCESS;
5364}
5365
5366
5367/** Opcode 0x0f 0x90. */
5368FNIEMOP_DEF(iemOp_seto_Eb)
5369{
5370 IEMOP_MNEMONIC(seto_Eb, "seto Eb");
5371 IEMOP_HLP_MIN_386();
5372 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5373
5374 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5375 * any way. AMD says it's "unused", whatever that means. We're
5376 * ignoring for now. */
5377 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5378 {
5379 /* register target */
5380 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5381 IEM_MC_BEGIN(0, 0);
5382 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
5383 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5384 } IEM_MC_ELSE() {
5385 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5386 } IEM_MC_ENDIF();
5387 IEM_MC_ADVANCE_RIP();
5388 IEM_MC_END();
5389 }
5390 else
5391 {
5392 /* memory target */
5393 IEM_MC_BEGIN(0, 1);
5394 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5395 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5396 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5397 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
5398 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5399 } IEM_MC_ELSE() {
5400 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5401 } IEM_MC_ENDIF();
5402 IEM_MC_ADVANCE_RIP();
5403 IEM_MC_END();
5404 }
5405 return VINF_SUCCESS;
5406}
5407
5408
5409/** Opcode 0x0f 0x91. */
5410FNIEMOP_DEF(iemOp_setno_Eb)
5411{
5412 IEMOP_MNEMONIC(setno_Eb, "setno Eb");
5413 IEMOP_HLP_MIN_386();
5414 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5415
5416 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5417 * any way. AMD says it's "unused", whatever that means. We're
5418 * ignoring for now. */
5419 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5420 {
5421 /* register target */
5422 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5423 IEM_MC_BEGIN(0, 0);
5424 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
5425 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5426 } IEM_MC_ELSE() {
5427 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5428 } IEM_MC_ENDIF();
5429 IEM_MC_ADVANCE_RIP();
5430 IEM_MC_END();
5431 }
5432 else
5433 {
5434 /* memory target */
5435 IEM_MC_BEGIN(0, 1);
5436 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5437 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5438 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5439 IEM_MC_IF_EFL_BIT_SET(X86_EFL_OF) {
5440 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5441 } IEM_MC_ELSE() {
5442 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5443 } IEM_MC_ENDIF();
5444 IEM_MC_ADVANCE_RIP();
5445 IEM_MC_END();
5446 }
5447 return VINF_SUCCESS;
5448}
5449
5450
5451/** Opcode 0x0f 0x92. */
5452FNIEMOP_DEF(iemOp_setc_Eb)
5453{
5454 IEMOP_MNEMONIC(setc_Eb, "setc Eb");
5455 IEMOP_HLP_MIN_386();
5456 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5457
5458 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5459 * any way. AMD says it's "unused", whatever that means. We're
5460 * ignoring for now. */
5461 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5462 {
5463 /* register target */
5464 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5465 IEM_MC_BEGIN(0, 0);
5466 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
5467 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5468 } IEM_MC_ELSE() {
5469 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5470 } IEM_MC_ENDIF();
5471 IEM_MC_ADVANCE_RIP();
5472 IEM_MC_END();
5473 }
5474 else
5475 {
5476 /* memory target */
5477 IEM_MC_BEGIN(0, 1);
5478 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5479 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5480 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5481 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
5482 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5483 } IEM_MC_ELSE() {
5484 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5485 } IEM_MC_ENDIF();
5486 IEM_MC_ADVANCE_RIP();
5487 IEM_MC_END();
5488 }
5489 return VINF_SUCCESS;
5490}
5491
5492
5493/** Opcode 0x0f 0x93. */
5494FNIEMOP_DEF(iemOp_setnc_Eb)
5495{
5496 IEMOP_MNEMONIC(setnc_Eb, "setnc Eb");
5497 IEMOP_HLP_MIN_386();
5498 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5499
5500 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5501 * any way. AMD says it's "unused", whatever that means. We're
5502 * ignoring for now. */
5503 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5504 {
5505 /* register target */
5506 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5507 IEM_MC_BEGIN(0, 0);
5508 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
5509 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5510 } IEM_MC_ELSE() {
5511 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5512 } IEM_MC_ENDIF();
5513 IEM_MC_ADVANCE_RIP();
5514 IEM_MC_END();
5515 }
5516 else
5517 {
5518 /* memory target */
5519 IEM_MC_BEGIN(0, 1);
5520 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5521 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5522 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5523 IEM_MC_IF_EFL_BIT_SET(X86_EFL_CF) {
5524 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5525 } IEM_MC_ELSE() {
5526 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5527 } IEM_MC_ENDIF();
5528 IEM_MC_ADVANCE_RIP();
5529 IEM_MC_END();
5530 }
5531 return VINF_SUCCESS;
5532}
5533
5534
5535/** Opcode 0x0f 0x94. */
5536FNIEMOP_DEF(iemOp_sete_Eb)
5537{
5538 IEMOP_MNEMONIC(sete_Eb, "sete Eb");
5539 IEMOP_HLP_MIN_386();
5540 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5541
5542 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5543 * any way. AMD says it's "unused", whatever that means. We're
5544 * ignoring for now. */
5545 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5546 {
5547 /* register target */
5548 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5549 IEM_MC_BEGIN(0, 0);
5550 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
5551 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5552 } IEM_MC_ELSE() {
5553 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5554 } IEM_MC_ENDIF();
5555 IEM_MC_ADVANCE_RIP();
5556 IEM_MC_END();
5557 }
5558 else
5559 {
5560 /* memory target */
5561 IEM_MC_BEGIN(0, 1);
5562 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5563 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5564 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5565 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
5566 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5567 } IEM_MC_ELSE() {
5568 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5569 } IEM_MC_ENDIF();
5570 IEM_MC_ADVANCE_RIP();
5571 IEM_MC_END();
5572 }
5573 return VINF_SUCCESS;
5574}
5575
5576
5577/** Opcode 0x0f 0x95. */
5578FNIEMOP_DEF(iemOp_setne_Eb)
5579{
5580 IEMOP_MNEMONIC(setne_Eb, "setne Eb");
5581 IEMOP_HLP_MIN_386();
5582 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5583
5584 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5585 * any way. AMD says it's "unused", whatever that means. We're
5586 * ignoring for now. */
5587 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5588 {
5589 /* register target */
5590 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5591 IEM_MC_BEGIN(0, 0);
5592 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
5593 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5594 } IEM_MC_ELSE() {
5595 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5596 } IEM_MC_ENDIF();
5597 IEM_MC_ADVANCE_RIP();
5598 IEM_MC_END();
5599 }
5600 else
5601 {
5602 /* memory target */
5603 IEM_MC_BEGIN(0, 1);
5604 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5605 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5606 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5607 IEM_MC_IF_EFL_BIT_SET(X86_EFL_ZF) {
5608 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5609 } IEM_MC_ELSE() {
5610 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5611 } IEM_MC_ENDIF();
5612 IEM_MC_ADVANCE_RIP();
5613 IEM_MC_END();
5614 }
5615 return VINF_SUCCESS;
5616}
5617
5618
5619/** Opcode 0x0f 0x96. */
5620FNIEMOP_DEF(iemOp_setbe_Eb)
5621{
5622 IEMOP_MNEMONIC(setbe_Eb, "setbe Eb");
5623 IEMOP_HLP_MIN_386();
5624 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5625
5626 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5627 * any way. AMD says it's "unused", whatever that means. We're
5628 * ignoring for now. */
5629 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5630 {
5631 /* register target */
5632 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5633 IEM_MC_BEGIN(0, 0);
5634 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5635 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5636 } IEM_MC_ELSE() {
5637 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5638 } IEM_MC_ENDIF();
5639 IEM_MC_ADVANCE_RIP();
5640 IEM_MC_END();
5641 }
5642 else
5643 {
5644 /* memory target */
5645 IEM_MC_BEGIN(0, 1);
5646 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5647 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5648 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5649 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5650 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5651 } IEM_MC_ELSE() {
5652 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5653 } IEM_MC_ENDIF();
5654 IEM_MC_ADVANCE_RIP();
5655 IEM_MC_END();
5656 }
5657 return VINF_SUCCESS;
5658}
5659
5660
5661/** Opcode 0x0f 0x97. */
5662FNIEMOP_DEF(iemOp_setnbe_Eb)
5663{
5664 IEMOP_MNEMONIC(setnbe_Eb, "setnbe Eb");
5665 IEMOP_HLP_MIN_386();
5666 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5667
5668 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5669 * any way. AMD says it's "unused", whatever that means. We're
5670 * ignoring for now. */
5671 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5672 {
5673 /* register target */
5674 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5675 IEM_MC_BEGIN(0, 0);
5676 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5677 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5678 } IEM_MC_ELSE() {
5679 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5680 } IEM_MC_ENDIF();
5681 IEM_MC_ADVANCE_RIP();
5682 IEM_MC_END();
5683 }
5684 else
5685 {
5686 /* memory target */
5687 IEM_MC_BEGIN(0, 1);
5688 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5689 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5690 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5691 IEM_MC_IF_EFL_ANY_BITS_SET(X86_EFL_CF | X86_EFL_ZF) {
5692 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5693 } IEM_MC_ELSE() {
5694 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5695 } IEM_MC_ENDIF();
5696 IEM_MC_ADVANCE_RIP();
5697 IEM_MC_END();
5698 }
5699 return VINF_SUCCESS;
5700}
5701
5702
5703/** Opcode 0x0f 0x98. */
5704FNIEMOP_DEF(iemOp_sets_Eb)
5705{
5706 IEMOP_MNEMONIC(sets_Eb, "sets Eb");
5707 IEMOP_HLP_MIN_386();
5708 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5709
5710 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5711 * any way. AMD says it's "unused", whatever that means. We're
5712 * ignoring for now. */
5713 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5714 {
5715 /* register target */
5716 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5717 IEM_MC_BEGIN(0, 0);
5718 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5719 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5720 } IEM_MC_ELSE() {
5721 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5722 } IEM_MC_ENDIF();
5723 IEM_MC_ADVANCE_RIP();
5724 IEM_MC_END();
5725 }
5726 else
5727 {
5728 /* memory target */
5729 IEM_MC_BEGIN(0, 1);
5730 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5731 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5732 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5733 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5734 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5735 } IEM_MC_ELSE() {
5736 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5737 } IEM_MC_ENDIF();
5738 IEM_MC_ADVANCE_RIP();
5739 IEM_MC_END();
5740 }
5741 return VINF_SUCCESS;
5742}
5743
5744
5745/** Opcode 0x0f 0x99. */
5746FNIEMOP_DEF(iemOp_setns_Eb)
5747{
5748 IEMOP_MNEMONIC(setns_Eb, "setns Eb");
5749 IEMOP_HLP_MIN_386();
5750 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5751
5752 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5753 * any way. AMD says it's "unused", whatever that means. We're
5754 * ignoring for now. */
5755 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5756 {
5757 /* register target */
5758 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5759 IEM_MC_BEGIN(0, 0);
5760 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5761 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5762 } IEM_MC_ELSE() {
5763 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5764 } IEM_MC_ENDIF();
5765 IEM_MC_ADVANCE_RIP();
5766 IEM_MC_END();
5767 }
5768 else
5769 {
5770 /* memory target */
5771 IEM_MC_BEGIN(0, 1);
5772 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5773 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5774 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5775 IEM_MC_IF_EFL_BIT_SET(X86_EFL_SF) {
5776 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5777 } IEM_MC_ELSE() {
5778 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5779 } IEM_MC_ENDIF();
5780 IEM_MC_ADVANCE_RIP();
5781 IEM_MC_END();
5782 }
5783 return VINF_SUCCESS;
5784}
5785
5786
5787/** Opcode 0x0f 0x9a. */
5788FNIEMOP_DEF(iemOp_setp_Eb)
5789{
5790 IEMOP_MNEMONIC(setp_Eb, "setp Eb");
5791 IEMOP_HLP_MIN_386();
5792 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5793
5794 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5795 * any way. AMD says it's "unused", whatever that means. We're
5796 * ignoring for now. */
5797 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5798 {
5799 /* register target */
5800 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5801 IEM_MC_BEGIN(0, 0);
5802 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5803 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5804 } IEM_MC_ELSE() {
5805 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5806 } IEM_MC_ENDIF();
5807 IEM_MC_ADVANCE_RIP();
5808 IEM_MC_END();
5809 }
5810 else
5811 {
5812 /* memory target */
5813 IEM_MC_BEGIN(0, 1);
5814 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5815 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5816 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5817 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5818 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5819 } IEM_MC_ELSE() {
5820 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5821 } IEM_MC_ENDIF();
5822 IEM_MC_ADVANCE_RIP();
5823 IEM_MC_END();
5824 }
5825 return VINF_SUCCESS;
5826}
5827
5828
5829/** Opcode 0x0f 0x9b. */
5830FNIEMOP_DEF(iemOp_setnp_Eb)
5831{
5832 IEMOP_MNEMONIC(setnp_Eb, "setnp Eb");
5833 IEMOP_HLP_MIN_386();
5834 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5835
5836 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5837 * any way. AMD says it's "unused", whatever that means. We're
5838 * ignoring for now. */
5839 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5840 {
5841 /* register target */
5842 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5843 IEM_MC_BEGIN(0, 0);
5844 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5845 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5846 } IEM_MC_ELSE() {
5847 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5848 } IEM_MC_ENDIF();
5849 IEM_MC_ADVANCE_RIP();
5850 IEM_MC_END();
5851 }
5852 else
5853 {
5854 /* memory target */
5855 IEM_MC_BEGIN(0, 1);
5856 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5857 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5858 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5859 IEM_MC_IF_EFL_BIT_SET(X86_EFL_PF) {
5860 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5861 } IEM_MC_ELSE() {
5862 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5863 } IEM_MC_ENDIF();
5864 IEM_MC_ADVANCE_RIP();
5865 IEM_MC_END();
5866 }
5867 return VINF_SUCCESS;
5868}
5869
5870
5871/** Opcode 0x0f 0x9c. */
5872FNIEMOP_DEF(iemOp_setl_Eb)
5873{
5874 IEMOP_MNEMONIC(setl_Eb, "setl Eb");
5875 IEMOP_HLP_MIN_386();
5876 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5877
5878 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5879 * any way. AMD says it's "unused", whatever that means. We're
5880 * ignoring for now. */
5881 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5882 {
5883 /* register target */
5884 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5885 IEM_MC_BEGIN(0, 0);
5886 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5887 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5888 } IEM_MC_ELSE() {
5889 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5890 } IEM_MC_ENDIF();
5891 IEM_MC_ADVANCE_RIP();
5892 IEM_MC_END();
5893 }
5894 else
5895 {
5896 /* memory target */
5897 IEM_MC_BEGIN(0, 1);
5898 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5899 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5900 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5901 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5902 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5903 } IEM_MC_ELSE() {
5904 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5905 } IEM_MC_ENDIF();
5906 IEM_MC_ADVANCE_RIP();
5907 IEM_MC_END();
5908 }
5909 return VINF_SUCCESS;
5910}
5911
5912
5913/** Opcode 0x0f 0x9d. */
5914FNIEMOP_DEF(iemOp_setnl_Eb)
5915{
5916 IEMOP_MNEMONIC(setnl_Eb, "setnl Eb");
5917 IEMOP_HLP_MIN_386();
5918 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5919
5920 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5921 * any way. AMD says it's "unused", whatever that means. We're
5922 * ignoring for now. */
5923 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5924 {
5925 /* register target */
5926 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5927 IEM_MC_BEGIN(0, 0);
5928 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5929 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5930 } IEM_MC_ELSE() {
5931 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5932 } IEM_MC_ENDIF();
5933 IEM_MC_ADVANCE_RIP();
5934 IEM_MC_END();
5935 }
5936 else
5937 {
5938 /* memory target */
5939 IEM_MC_BEGIN(0, 1);
5940 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5941 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5942 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5943 IEM_MC_IF_EFL_BITS_NE(X86_EFL_SF, X86_EFL_OF) {
5944 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5945 } IEM_MC_ELSE() {
5946 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5947 } IEM_MC_ENDIF();
5948 IEM_MC_ADVANCE_RIP();
5949 IEM_MC_END();
5950 }
5951 return VINF_SUCCESS;
5952}
5953
5954
5955/** Opcode 0x0f 0x9e. */
5956FNIEMOP_DEF(iemOp_setle_Eb)
5957{
5958 IEMOP_MNEMONIC(setle_Eb, "setle Eb");
5959 IEMOP_HLP_MIN_386();
5960 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
5961
5962 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
5963 * any way. AMD says it's "unused", whatever that means. We're
5964 * ignoring for now. */
5965 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
5966 {
5967 /* register target */
5968 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5969 IEM_MC_BEGIN(0, 0);
5970 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5971 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
5972 } IEM_MC_ELSE() {
5973 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
5974 } IEM_MC_ENDIF();
5975 IEM_MC_ADVANCE_RIP();
5976 IEM_MC_END();
5977 }
5978 else
5979 {
5980 /* memory target */
5981 IEM_MC_BEGIN(0, 1);
5982 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
5983 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
5984 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
5985 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
5986 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
5987 } IEM_MC_ELSE() {
5988 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
5989 } IEM_MC_ENDIF();
5990 IEM_MC_ADVANCE_RIP();
5991 IEM_MC_END();
5992 }
5993 return VINF_SUCCESS;
5994}
5995
5996
5997/** Opcode 0x0f 0x9f. */
5998FNIEMOP_DEF(iemOp_setnle_Eb)
5999{
6000 IEMOP_MNEMONIC(setnle_Eb, "setnle Eb");
6001 IEMOP_HLP_MIN_386();
6002 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
6003
6004 /** @todo Encoding test: Check if the 'reg' field is ignored or decoded in
6005 * any way. AMD says it's "unused", whatever that means. We're
6006 * ignoring for now. */
6007 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
6008 {
6009 /* register target */
6010 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6011 IEM_MC_BEGIN(0, 0);
6012 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
6013 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 0);
6014 } IEM_MC_ELSE() {
6015 IEM_MC_STORE_GREG_U8_CONST((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, 1);
6016 } IEM_MC_ENDIF();
6017 IEM_MC_ADVANCE_RIP();
6018 IEM_MC_END();
6019 }
6020 else
6021 {
6022 /* memory target */
6023 IEM_MC_BEGIN(0, 1);
6024 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6025 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6026 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6027 IEM_MC_IF_EFL_BIT_SET_OR_BITS_NE(X86_EFL_ZF, X86_EFL_SF, X86_EFL_OF) {
6028 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6029 } IEM_MC_ELSE() {
6030 IEM_MC_STORE_MEM_U8_CONST(pVCpu->iem.s.iEffSeg, GCPtrEffDst, 1);
6031 } IEM_MC_ENDIF();
6032 IEM_MC_ADVANCE_RIP();
6033 IEM_MC_END();
6034 }
6035 return VINF_SUCCESS;
6036}
6037
6038
6039/**
6040 * Common 'push segment-register' helper.
6041 */
6042FNIEMOP_DEF_1(iemOpCommonPushSReg, uint8_t, iReg)
6043{
6044 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6045 Assert(iReg < X86_SREG_FS || pVCpu->iem.s.enmCpuMode != IEMMODE_64BIT);
6046 IEMOP_HLP_DEFAULT_64BIT_OP_SIZE();
6047
6048 switch (pVCpu->iem.s.enmEffOpSize)
6049 {
6050 case IEMMODE_16BIT:
6051 IEM_MC_BEGIN(0, 1);
6052 IEM_MC_LOCAL(uint16_t, u16Value);
6053 IEM_MC_FETCH_SREG_U16(u16Value, iReg);
6054 IEM_MC_PUSH_U16(u16Value);
6055 IEM_MC_ADVANCE_RIP();
6056 IEM_MC_END();
6057 break;
6058
6059 case IEMMODE_32BIT:
6060 IEM_MC_BEGIN(0, 1);
6061 IEM_MC_LOCAL(uint32_t, u32Value);
6062 IEM_MC_FETCH_SREG_ZX_U32(u32Value, iReg);
6063 IEM_MC_PUSH_U32_SREG(u32Value);
6064 IEM_MC_ADVANCE_RIP();
6065 IEM_MC_END();
6066 break;
6067
6068 case IEMMODE_64BIT:
6069 IEM_MC_BEGIN(0, 1);
6070 IEM_MC_LOCAL(uint64_t, u64Value);
6071 IEM_MC_FETCH_SREG_ZX_U64(u64Value, iReg);
6072 IEM_MC_PUSH_U64(u64Value);
6073 IEM_MC_ADVANCE_RIP();
6074 IEM_MC_END();
6075 break;
6076 }
6077
6078 return VINF_SUCCESS;
6079}
6080
6081
6082/** Opcode 0x0f 0xa0. */
6083FNIEMOP_DEF(iemOp_push_fs)
6084{
6085 IEMOP_MNEMONIC(push_fs, "push fs");
6086 IEMOP_HLP_MIN_386();
6087 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6088 return FNIEMOP_CALL_1(iemOpCommonPushSReg, X86_SREG_FS);
6089}
6090
6091
6092/** Opcode 0x0f 0xa1. */
6093FNIEMOP_DEF(iemOp_pop_fs)
6094{
6095 IEMOP_MNEMONIC(pop_fs, "pop fs");
6096 IEMOP_HLP_MIN_386();
6097 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6098 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_pop_Sreg, X86_SREG_FS, pVCpu->iem.s.enmEffOpSize);
6099}
6100
6101
6102/** Opcode 0x0f 0xa2. */
6103FNIEMOP_DEF(iemOp_cpuid)
6104{
6105 IEMOP_MNEMONIC(cpuid, "cpuid");
6106 IEMOP_HLP_MIN_486(); /* not all 486es. */
6107 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6108 return IEM_MC_DEFER_TO_CIMPL_0(iemCImpl_cpuid);
6109}
6110
6111
6112/**
6113 * Common worker for iemOp_bt_Ev_Gv, iemOp_btc_Ev_Gv, iemOp_btr_Ev_Gv and
6114 * iemOp_bts_Ev_Gv.
6115 */
6116FNIEMOP_DEF_1(iemOpCommonBit_Ev_Gv, PCIEMOPBINSIZES, pImpl)
6117{
6118 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
6119 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF);
6120
6121 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
6122 {
6123 /* register destination. */
6124 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6125 switch (pVCpu->iem.s.enmEffOpSize)
6126 {
6127 case IEMMODE_16BIT:
6128 IEM_MC_BEGIN(3, 0);
6129 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6130 IEM_MC_ARG(uint16_t, u16Src, 1);
6131 IEM_MC_ARG(uint32_t *, pEFlags, 2);
6132
6133 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6134 IEM_MC_AND_LOCAL_U16(u16Src, 0xf);
6135 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6136 IEM_MC_REF_EFLAGS(pEFlags);
6137 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU16, pu16Dst, u16Src, pEFlags);
6138
6139 IEM_MC_ADVANCE_RIP();
6140 IEM_MC_END();
6141 return VINF_SUCCESS;
6142
6143 case IEMMODE_32BIT:
6144 IEM_MC_BEGIN(3, 0);
6145 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6146 IEM_MC_ARG(uint32_t, u32Src, 1);
6147 IEM_MC_ARG(uint32_t *, pEFlags, 2);
6148
6149 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6150 IEM_MC_AND_LOCAL_U32(u32Src, 0x1f);
6151 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6152 IEM_MC_REF_EFLAGS(pEFlags);
6153 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU32, pu32Dst, u32Src, pEFlags);
6154
6155 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
6156 IEM_MC_ADVANCE_RIP();
6157 IEM_MC_END();
6158 return VINF_SUCCESS;
6159
6160 case IEMMODE_64BIT:
6161 IEM_MC_BEGIN(3, 0);
6162 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6163 IEM_MC_ARG(uint64_t, u64Src, 1);
6164 IEM_MC_ARG(uint32_t *, pEFlags, 2);
6165
6166 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6167 IEM_MC_AND_LOCAL_U64(u64Src, 0x3f);
6168 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6169 IEM_MC_REF_EFLAGS(pEFlags);
6170 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU64, pu64Dst, u64Src, pEFlags);
6171
6172 IEM_MC_ADVANCE_RIP();
6173 IEM_MC_END();
6174 return VINF_SUCCESS;
6175
6176 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6177 }
6178 }
6179 else
6180 {
6181 /* memory destination. */
6182
6183 uint32_t fAccess;
6184 if (pImpl->pfnLockedU16)
6185 fAccess = IEM_ACCESS_DATA_RW;
6186 else /* BT */
6187 fAccess = IEM_ACCESS_DATA_R;
6188
6189 /** @todo test negative bit offsets! */
6190 switch (pVCpu->iem.s.enmEffOpSize)
6191 {
6192 case IEMMODE_16BIT:
6193 IEM_MC_BEGIN(3, 2);
6194 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6195 IEM_MC_ARG(uint16_t, u16Src, 1);
6196 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
6197 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6198 IEM_MC_LOCAL(int16_t, i16AddrAdj);
6199
6200 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6201 if (pImpl->pfnLockedU16)
6202 IEMOP_HLP_DONE_DECODING();
6203 else
6204 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6205 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6206 IEM_MC_ASSIGN(i16AddrAdj, u16Src);
6207 IEM_MC_AND_ARG_U16(u16Src, 0x0f);
6208 IEM_MC_SAR_LOCAL_S16(i16AddrAdj, 4);
6209 IEM_MC_SHL_LOCAL_S16(i16AddrAdj, 1);
6210 IEM_MC_ADD_LOCAL_S16_TO_EFF_ADDR(GCPtrEffDst, i16AddrAdj);
6211 IEM_MC_FETCH_EFLAGS(EFlags);
6212
6213 IEM_MC_MEM_MAP(pu16Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6214 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
6215 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU16, pu16Dst, u16Src, pEFlags);
6216 else
6217 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU16, pu16Dst, u16Src, pEFlags);
6218 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, fAccess);
6219
6220 IEM_MC_COMMIT_EFLAGS(EFlags);
6221 IEM_MC_ADVANCE_RIP();
6222 IEM_MC_END();
6223 return VINF_SUCCESS;
6224
6225 case IEMMODE_32BIT:
6226 IEM_MC_BEGIN(3, 2);
6227 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6228 IEM_MC_ARG(uint32_t, u32Src, 1);
6229 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
6230 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6231 IEM_MC_LOCAL(int32_t, i32AddrAdj);
6232
6233 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6234 if (pImpl->pfnLockedU16)
6235 IEMOP_HLP_DONE_DECODING();
6236 else
6237 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6238 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6239 IEM_MC_ASSIGN(i32AddrAdj, u32Src);
6240 IEM_MC_AND_ARG_U32(u32Src, 0x1f);
6241 IEM_MC_SAR_LOCAL_S32(i32AddrAdj, 5);
6242 IEM_MC_SHL_LOCAL_S32(i32AddrAdj, 2);
6243 IEM_MC_ADD_LOCAL_S32_TO_EFF_ADDR(GCPtrEffDst, i32AddrAdj);
6244 IEM_MC_FETCH_EFLAGS(EFlags);
6245
6246 IEM_MC_MEM_MAP(pu32Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6247 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
6248 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU32, pu32Dst, u32Src, pEFlags);
6249 else
6250 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU32, pu32Dst, u32Src, pEFlags);
6251 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, fAccess);
6252
6253 IEM_MC_COMMIT_EFLAGS(EFlags);
6254 IEM_MC_ADVANCE_RIP();
6255 IEM_MC_END();
6256 return VINF_SUCCESS;
6257
6258 case IEMMODE_64BIT:
6259 IEM_MC_BEGIN(3, 2);
6260 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6261 IEM_MC_ARG(uint64_t, u64Src, 1);
6262 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
6263 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6264 IEM_MC_LOCAL(int64_t, i64AddrAdj);
6265
6266 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6267 if (pImpl->pfnLockedU16)
6268 IEMOP_HLP_DONE_DECODING();
6269 else
6270 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6271 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6272 IEM_MC_ASSIGN(i64AddrAdj, u64Src);
6273 IEM_MC_AND_ARG_U64(u64Src, 0x3f);
6274 IEM_MC_SAR_LOCAL_S64(i64AddrAdj, 6);
6275 IEM_MC_SHL_LOCAL_S64(i64AddrAdj, 3);
6276 IEM_MC_ADD_LOCAL_S64_TO_EFF_ADDR(GCPtrEffDst, i64AddrAdj);
6277 IEM_MC_FETCH_EFLAGS(EFlags);
6278
6279 IEM_MC_MEM_MAP(pu64Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6280 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
6281 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU64, pu64Dst, u64Src, pEFlags);
6282 else
6283 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU64, pu64Dst, u64Src, pEFlags);
6284 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, fAccess);
6285
6286 IEM_MC_COMMIT_EFLAGS(EFlags);
6287 IEM_MC_ADVANCE_RIP();
6288 IEM_MC_END();
6289 return VINF_SUCCESS;
6290
6291 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6292 }
6293 }
6294}
6295
6296
6297/** Opcode 0x0f 0xa3. */
6298FNIEMOP_DEF(iemOp_bt_Ev_Gv)
6299{
6300 IEMOP_MNEMONIC(bt_Ev_Gv, "bt Ev,Gv");
6301 IEMOP_HLP_MIN_386();
6302 return FNIEMOP_CALL_1(iemOpCommonBit_Ev_Gv, &g_iemAImpl_bt);
6303}
6304
6305
6306/**
6307 * Common worker for iemOp_shrd_Ev_Gv_Ib and iemOp_shld_Ev_Gv_Ib.
6308 */
6309FNIEMOP_DEF_1(iemOpCommonShldShrd_Ib, PCIEMOPSHIFTDBLSIZES, pImpl)
6310{
6311 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
6312 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_AF | X86_EFL_OF);
6313
6314 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
6315 {
6316 uint8_t cShift; IEM_OPCODE_GET_NEXT_U8(&cShift);
6317 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6318
6319 switch (pVCpu->iem.s.enmEffOpSize)
6320 {
6321 case IEMMODE_16BIT:
6322 IEM_MC_BEGIN(4, 0);
6323 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6324 IEM_MC_ARG(uint16_t, u16Src, 1);
6325 IEM_MC_ARG_CONST(uint8_t, cShiftArg, /*=*/cShift, 2);
6326 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6327
6328 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6329 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6330 IEM_MC_REF_EFLAGS(pEFlags);
6331 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU16, pu16Dst, u16Src, cShiftArg, pEFlags);
6332
6333 IEM_MC_ADVANCE_RIP();
6334 IEM_MC_END();
6335 return VINF_SUCCESS;
6336
6337 case IEMMODE_32BIT:
6338 IEM_MC_BEGIN(4, 0);
6339 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6340 IEM_MC_ARG(uint32_t, u32Src, 1);
6341 IEM_MC_ARG_CONST(uint8_t, cShiftArg, /*=*/cShift, 2);
6342 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6343
6344 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6345 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6346 IEM_MC_REF_EFLAGS(pEFlags);
6347 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU32, pu32Dst, u32Src, cShiftArg, pEFlags);
6348
6349 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
6350 IEM_MC_ADVANCE_RIP();
6351 IEM_MC_END();
6352 return VINF_SUCCESS;
6353
6354 case IEMMODE_64BIT:
6355 IEM_MC_BEGIN(4, 0);
6356 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6357 IEM_MC_ARG(uint64_t, u64Src, 1);
6358 IEM_MC_ARG_CONST(uint8_t, cShiftArg, /*=*/cShift, 2);
6359 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6360
6361 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6362 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6363 IEM_MC_REF_EFLAGS(pEFlags);
6364 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU64, pu64Dst, u64Src, cShiftArg, pEFlags);
6365
6366 IEM_MC_ADVANCE_RIP();
6367 IEM_MC_END();
6368 return VINF_SUCCESS;
6369
6370 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6371 }
6372 }
6373 else
6374 {
6375 switch (pVCpu->iem.s.enmEffOpSize)
6376 {
6377 case IEMMODE_16BIT:
6378 IEM_MC_BEGIN(4, 2);
6379 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6380 IEM_MC_ARG(uint16_t, u16Src, 1);
6381 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6382 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6383 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6384
6385 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
6386 uint8_t cShift; IEM_OPCODE_GET_NEXT_U8(&cShift);
6387 IEM_MC_ASSIGN(cShiftArg, cShift);
6388 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6389 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6390 IEM_MC_FETCH_EFLAGS(EFlags);
6391 IEM_MC_MEM_MAP(pu16Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6392 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU16, pu16Dst, u16Src, cShiftArg, pEFlags);
6393
6394 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, IEM_ACCESS_DATA_RW);
6395 IEM_MC_COMMIT_EFLAGS(EFlags);
6396 IEM_MC_ADVANCE_RIP();
6397 IEM_MC_END();
6398 return VINF_SUCCESS;
6399
6400 case IEMMODE_32BIT:
6401 IEM_MC_BEGIN(4, 2);
6402 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6403 IEM_MC_ARG(uint32_t, u32Src, 1);
6404 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6405 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6406 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6407
6408 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
6409 uint8_t cShift; IEM_OPCODE_GET_NEXT_U8(&cShift);
6410 IEM_MC_ASSIGN(cShiftArg, cShift);
6411 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6412 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6413 IEM_MC_FETCH_EFLAGS(EFlags);
6414 IEM_MC_MEM_MAP(pu32Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6415 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU32, pu32Dst, u32Src, cShiftArg, pEFlags);
6416
6417 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, IEM_ACCESS_DATA_RW);
6418 IEM_MC_COMMIT_EFLAGS(EFlags);
6419 IEM_MC_ADVANCE_RIP();
6420 IEM_MC_END();
6421 return VINF_SUCCESS;
6422
6423 case IEMMODE_64BIT:
6424 IEM_MC_BEGIN(4, 2);
6425 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6426 IEM_MC_ARG(uint64_t, u64Src, 1);
6427 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6428 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6429 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6430
6431 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
6432 uint8_t cShift; IEM_OPCODE_GET_NEXT_U8(&cShift);
6433 IEM_MC_ASSIGN(cShiftArg, cShift);
6434 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6435 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6436 IEM_MC_FETCH_EFLAGS(EFlags);
6437 IEM_MC_MEM_MAP(pu64Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6438 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU64, pu64Dst, u64Src, cShiftArg, pEFlags);
6439
6440 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, IEM_ACCESS_DATA_RW);
6441 IEM_MC_COMMIT_EFLAGS(EFlags);
6442 IEM_MC_ADVANCE_RIP();
6443 IEM_MC_END();
6444 return VINF_SUCCESS;
6445
6446 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6447 }
6448 }
6449}
6450
6451
6452/**
6453 * Common worker for iemOp_shrd_Ev_Gv_CL and iemOp_shld_Ev_Gv_CL.
6454 */
6455FNIEMOP_DEF_1(iemOpCommonShldShrd_CL, PCIEMOPSHIFTDBLSIZES, pImpl)
6456{
6457 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
6458 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_AF | X86_EFL_OF);
6459
6460 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
6461 {
6462 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6463
6464 switch (pVCpu->iem.s.enmEffOpSize)
6465 {
6466 case IEMMODE_16BIT:
6467 IEM_MC_BEGIN(4, 0);
6468 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6469 IEM_MC_ARG(uint16_t, u16Src, 1);
6470 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6471 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6472
6473 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6474 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6475 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6476 IEM_MC_REF_EFLAGS(pEFlags);
6477 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU16, pu16Dst, u16Src, cShiftArg, pEFlags);
6478
6479 IEM_MC_ADVANCE_RIP();
6480 IEM_MC_END();
6481 return VINF_SUCCESS;
6482
6483 case IEMMODE_32BIT:
6484 IEM_MC_BEGIN(4, 0);
6485 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6486 IEM_MC_ARG(uint32_t, u32Src, 1);
6487 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6488 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6489
6490 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6491 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6492 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6493 IEM_MC_REF_EFLAGS(pEFlags);
6494 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU32, pu32Dst, u32Src, cShiftArg, pEFlags);
6495
6496 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
6497 IEM_MC_ADVANCE_RIP();
6498 IEM_MC_END();
6499 return VINF_SUCCESS;
6500
6501 case IEMMODE_64BIT:
6502 IEM_MC_BEGIN(4, 0);
6503 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6504 IEM_MC_ARG(uint64_t, u64Src, 1);
6505 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6506 IEM_MC_ARG(uint32_t *, pEFlags, 3);
6507
6508 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6509 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
6510 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6511 IEM_MC_REF_EFLAGS(pEFlags);
6512 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU64, pu64Dst, u64Src, cShiftArg, pEFlags);
6513
6514 IEM_MC_ADVANCE_RIP();
6515 IEM_MC_END();
6516 return VINF_SUCCESS;
6517
6518 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6519 }
6520 }
6521 else
6522 {
6523 switch (pVCpu->iem.s.enmEffOpSize)
6524 {
6525 case IEMMODE_16BIT:
6526 IEM_MC_BEGIN(4, 2);
6527 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
6528 IEM_MC_ARG(uint16_t, u16Src, 1);
6529 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6530 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6531 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6532
6533 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6534 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6535 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6536 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6537 IEM_MC_FETCH_EFLAGS(EFlags);
6538 IEM_MC_MEM_MAP(pu16Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6539 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU16, pu16Dst, u16Src, cShiftArg, pEFlags);
6540
6541 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, IEM_ACCESS_DATA_RW);
6542 IEM_MC_COMMIT_EFLAGS(EFlags);
6543 IEM_MC_ADVANCE_RIP();
6544 IEM_MC_END();
6545 return VINF_SUCCESS;
6546
6547 case IEMMODE_32BIT:
6548 IEM_MC_BEGIN(4, 2);
6549 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
6550 IEM_MC_ARG(uint32_t, u32Src, 1);
6551 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6552 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6553 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6554
6555 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6556 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6557 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6558 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6559 IEM_MC_FETCH_EFLAGS(EFlags);
6560 IEM_MC_MEM_MAP(pu32Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6561 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU32, pu32Dst, u32Src, cShiftArg, pEFlags);
6562
6563 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, IEM_ACCESS_DATA_RW);
6564 IEM_MC_COMMIT_EFLAGS(EFlags);
6565 IEM_MC_ADVANCE_RIP();
6566 IEM_MC_END();
6567 return VINF_SUCCESS;
6568
6569 case IEMMODE_64BIT:
6570 IEM_MC_BEGIN(4, 2);
6571 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
6572 IEM_MC_ARG(uint64_t, u64Src, 1);
6573 IEM_MC_ARG(uint8_t, cShiftArg, 2);
6574 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
6575 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
6576
6577 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
6578 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6579 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
6580 IEM_MC_FETCH_GREG_U8(cShiftArg, X86_GREG_xCX);
6581 IEM_MC_FETCH_EFLAGS(EFlags);
6582 IEM_MC_MEM_MAP(pu64Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
6583 IEM_MC_CALL_VOID_AIMPL_4(pImpl->pfnNormalU64, pu64Dst, u64Src, cShiftArg, pEFlags);
6584
6585 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, IEM_ACCESS_DATA_RW);
6586 IEM_MC_COMMIT_EFLAGS(EFlags);
6587 IEM_MC_ADVANCE_RIP();
6588 IEM_MC_END();
6589 return VINF_SUCCESS;
6590
6591 IEM_NOT_REACHED_DEFAULT_CASE_RET();
6592 }
6593 }
6594}
6595
6596
6597
6598/** Opcode 0x0f 0xa4. */
6599FNIEMOP_DEF(iemOp_shld_Ev_Gv_Ib)
6600{
6601 IEMOP_MNEMONIC(shld_Ev_Gv_Ib, "shld Ev,Gv,Ib");
6602 IEMOP_HLP_MIN_386();
6603 return FNIEMOP_CALL_1(iemOpCommonShldShrd_Ib, &g_iemAImpl_shld);
6604}
6605
6606
6607/** Opcode 0x0f 0xa5. */
6608FNIEMOP_DEF(iemOp_shld_Ev_Gv_CL)
6609{
6610 IEMOP_MNEMONIC(shld_Ev_Gv_CL, "shld Ev,Gv,CL");
6611 IEMOP_HLP_MIN_386();
6612 return FNIEMOP_CALL_1(iemOpCommonShldShrd_CL, &g_iemAImpl_shld);
6613}
6614
6615
6616/** Opcode 0x0f 0xa8. */
6617FNIEMOP_DEF(iemOp_push_gs)
6618{
6619 IEMOP_MNEMONIC(push_gs, "push gs");
6620 IEMOP_HLP_MIN_386();
6621 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6622 return FNIEMOP_CALL_1(iemOpCommonPushSReg, X86_SREG_GS);
6623}
6624
6625
6626/** Opcode 0x0f 0xa9. */
6627FNIEMOP_DEF(iemOp_pop_gs)
6628{
6629 IEMOP_MNEMONIC(pop_gs, "pop gs");
6630 IEMOP_HLP_MIN_386();
6631 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6632 return IEM_MC_DEFER_TO_CIMPL_2(iemCImpl_pop_Sreg, X86_SREG_GS, pVCpu->iem.s.enmEffOpSize);
6633}
6634
6635
6636/** Opcode 0x0f 0xaa. */
6637FNIEMOP_DEF(iemOp_rsm)
6638{
6639 IEMOP_MNEMONIC(rsm, "rsm");
6640 IEMOP_HLP_SVM_CTRL_INTERCEPT(pVCpu, SVM_CTRL_INTERCEPT_RSM, SVM_EXIT_RSM, 0, 0);
6641 /** @todo rsm - for the regular case (above handles only the SVM nested-guest
6642 * intercept). */
6643 IEMOP_BITCH_ABOUT_STUB();
6644 return IEMOP_RAISE_INVALID_OPCODE();
6645}
6646
6647//IEMOP_HLP_MIN_386();
6648
6649
6650/** Opcode 0x0f 0xab. */
6651FNIEMOP_DEF(iemOp_bts_Ev_Gv)
6652{
6653 IEMOP_MNEMONIC(bts_Ev_Gv, "bts Ev,Gv");
6654 IEMOP_HLP_MIN_386();
6655 return FNIEMOP_CALL_1(iemOpCommonBit_Ev_Gv, &g_iemAImpl_bts);
6656}
6657
6658
6659/** Opcode 0x0f 0xac. */
6660FNIEMOP_DEF(iemOp_shrd_Ev_Gv_Ib)
6661{
6662 IEMOP_MNEMONIC(shrd_Ev_Gv_Ib, "shrd Ev,Gv,Ib");
6663 IEMOP_HLP_MIN_386();
6664 return FNIEMOP_CALL_1(iemOpCommonShldShrd_Ib, &g_iemAImpl_shrd);
6665}
6666
6667
6668/** Opcode 0x0f 0xad. */
6669FNIEMOP_DEF(iemOp_shrd_Ev_Gv_CL)
6670{
6671 IEMOP_MNEMONIC(shrd_Ev_Gv_CL, "shrd Ev,Gv,CL");
6672 IEMOP_HLP_MIN_386();
6673 return FNIEMOP_CALL_1(iemOpCommonShldShrd_CL, &g_iemAImpl_shrd);
6674}
6675
6676
6677/** Opcode 0x0f 0xae mem/0. */
6678FNIEMOP_DEF_1(iemOp_Grp15_fxsave, uint8_t, bRm)
6679{
6680 IEMOP_MNEMONIC(fxsave, "fxsave m512");
6681 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fFxSaveRstor)
6682 return IEMOP_RAISE_INVALID_OPCODE();
6683
6684 IEM_MC_BEGIN(3, 1);
6685 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6686 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6687 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 2);
6688 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6689 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6690 IEM_MC_ACTUALIZE_FPU_STATE_FOR_READ();
6691 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6692 IEM_MC_CALL_CIMPL_3(iemCImpl_fxsave, iEffSeg, GCPtrEff, enmEffOpSize);
6693 IEM_MC_END();
6694 return VINF_SUCCESS;
6695}
6696
6697
6698/** Opcode 0x0f 0xae mem/1. */
6699FNIEMOP_DEF_1(iemOp_Grp15_fxrstor, uint8_t, bRm)
6700{
6701 IEMOP_MNEMONIC(fxrstor, "fxrstor m512");
6702 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fFxSaveRstor)
6703 return IEMOP_RAISE_INVALID_OPCODE();
6704
6705 IEM_MC_BEGIN(3, 1);
6706 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6707 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6708 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 2);
6709 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6710 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6711 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
6712 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6713 IEM_MC_CALL_CIMPL_3(iemCImpl_fxrstor, iEffSeg, GCPtrEff, enmEffOpSize);
6714 IEM_MC_END();
6715 return VINF_SUCCESS;
6716}
6717
6718
6719/**
6720 * @opmaps grp15
6721 * @opcode !11/2
6722 * @oppfx none
6723 * @opcpuid sse
6724 * @opgroup og_sse_mxcsrsm
6725 * @opxcpttype 5
6726 * @optest op1=0 -> mxcsr=0
6727 * @optest op1=0x2083 -> mxcsr=0x2083
6728 * @optest op1=0xfffffffe -> value.xcpt=0xd
6729 * @optest op1=0x2083 cr0|=ts -> value.xcpt=0x7
6730 * @optest op1=0x2083 cr0|=em -> value.xcpt=0x6
6731 * @optest op1=0x2083 cr0|=mp -> mxcsr=0x2083
6732 * @optest op1=0x2083 cr4&~=osfxsr -> value.xcpt=0x6
6733 * @optest op1=0x2083 cr0|=ts,em -> value.xcpt=0x6
6734 * @optest op1=0x2083 cr0|=em cr4&~=osfxsr -> value.xcpt=0x6
6735 * @optest op1=0x2083 cr0|=ts,em cr4&~=osfxsr -> value.xcpt=0x6
6736 * @optest op1=0x2083 cr0|=ts,em,mp cr4&~=osfxsr -> value.xcpt=0x6
6737 */
6738FNIEMOP_DEF_1(iemOp_Grp15_ldmxcsr, uint8_t, bRm)
6739{
6740 IEMOP_MNEMONIC1(M_MEM, LDMXCSR, ldmxcsr, Md_RO, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
6741 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse)
6742 return IEMOP_RAISE_INVALID_OPCODE();
6743
6744 IEM_MC_BEGIN(2, 0);
6745 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6746 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6747 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6748 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6749 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
6750 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6751 IEM_MC_CALL_CIMPL_2(iemCImpl_ldmxcsr, iEffSeg, GCPtrEff);
6752 IEM_MC_END();
6753 return VINF_SUCCESS;
6754}
6755
6756
6757/**
6758 * @opmaps grp15
6759 * @opcode !11/3
6760 * @oppfx none
6761 * @opcpuid sse
6762 * @opgroup og_sse_mxcsrsm
6763 * @opxcpttype 5
6764 * @optest mxcsr=0 -> op1=0
6765 * @optest mxcsr=0x2083 -> op1=0x2083
6766 * @optest mxcsr=0x2084 cr0|=ts -> value.xcpt=0x7
6767 * @optest mxcsr=0x2085 cr0|=em -> value.xcpt=0x6
6768 * @optest mxcsr=0x2086 cr0|=mp -> op1=0x2086
6769 * @optest mxcsr=0x2087 cr4&~=osfxsr -> value.xcpt=0x6
6770 * @optest mxcsr=0x2088 cr0|=ts,em -> value.xcpt=0x6
6771 * @optest mxcsr=0x2089 cr0|=em cr4&~=osfxsr -> value.xcpt=0x6
6772 * @optest mxcsr=0x208a cr0|=ts,em cr4&~=osfxsr -> value.xcpt=0x6
6773 * @optest mxcsr=0x208b cr0|=ts,em,mp cr4&~=osfxsr -> value.xcpt=0x6
6774 */
6775FNIEMOP_DEF_1(iemOp_Grp15_stmxcsr, uint8_t, bRm)
6776{
6777 IEMOP_MNEMONIC1(M_MEM, STMXCSR, stmxcsr, Md_WO, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
6778 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse)
6779 return IEMOP_RAISE_INVALID_OPCODE();
6780
6781 IEM_MC_BEGIN(2, 0);
6782 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6783 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6784 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6785 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6786 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
6787 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6788 IEM_MC_CALL_CIMPL_2(iemCImpl_stmxcsr, iEffSeg, GCPtrEff);
6789 IEM_MC_END();
6790 return VINF_SUCCESS;
6791}
6792
6793
6794/**
6795 * @opmaps grp15
6796 * @opcode !11/4
6797 * @oppfx none
6798 * @opcpuid xsave
6799 * @opgroup og_system
6800 * @opxcpttype none
6801 */
6802FNIEMOP_DEF_1(iemOp_Grp15_xsave, uint8_t, bRm)
6803{
6804 IEMOP_MNEMONIC1(M_MEM, XSAVE, xsave, M_RW, DISOPTYPE_HARMLESS, 0);
6805 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fXSaveRstor)
6806 return IEMOP_RAISE_INVALID_OPCODE();
6807
6808 IEM_MC_BEGIN(3, 0);
6809 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6810 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6811 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 2);
6812 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6813 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6814 IEM_MC_ACTUALIZE_FPU_STATE_FOR_READ();
6815 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6816 IEM_MC_CALL_CIMPL_3(iemCImpl_xsave, iEffSeg, GCPtrEff, enmEffOpSize);
6817 IEM_MC_END();
6818 return VINF_SUCCESS;
6819}
6820
6821
6822/**
6823 * @opmaps grp15
6824 * @opcode !11/5
6825 * @oppfx none
6826 * @opcpuid xsave
6827 * @opgroup og_system
6828 * @opxcpttype none
6829 */
6830FNIEMOP_DEF_1(iemOp_Grp15_xrstor, uint8_t, bRm)
6831{
6832 IEMOP_MNEMONIC1(M_MEM, XRSTOR, xrstor, M_RO, DISOPTYPE_HARMLESS, 0);
6833 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fXSaveRstor)
6834 return IEMOP_RAISE_INVALID_OPCODE();
6835
6836 IEM_MC_BEGIN(3, 0);
6837 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6838 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6839 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 2);
6840 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6841 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6842 IEM_MC_ACTUALIZE_FPU_STATE_FOR_READ();
6843 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6844 IEM_MC_CALL_CIMPL_3(iemCImpl_xrstor, iEffSeg, GCPtrEff, enmEffOpSize);
6845 IEM_MC_END();
6846 return VINF_SUCCESS;
6847}
6848
6849/** Opcode 0x0f 0xae mem/6. */
6850FNIEMOP_UD_STUB_1(iemOp_Grp15_xsaveopt, uint8_t, bRm);
6851
6852/**
6853 * @opmaps grp15
6854 * @opcode !11/7
6855 * @oppfx none
6856 * @opcpuid clfsh
6857 * @opgroup og_cachectl
6858 * @optest op1=1 ->
6859 */
6860FNIEMOP_DEF_1(iemOp_Grp15_clflush, uint8_t, bRm)
6861{
6862 IEMOP_MNEMONIC1(M_MEM, CLFLUSH, clflush, Mb_RO, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
6863 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fClFlush)
6864 return FNIEMOP_CALL_1(iemOp_InvalidWithRMAllNeeded, bRm);
6865
6866 IEM_MC_BEGIN(2, 0);
6867 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6868 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6869 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6870 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6871 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6872 IEM_MC_CALL_CIMPL_2(iemCImpl_clflush_clflushopt, iEffSeg, GCPtrEff);
6873 IEM_MC_END();
6874 return VINF_SUCCESS;
6875}
6876
6877/**
6878 * @opmaps grp15
6879 * @opcode !11/7
6880 * @oppfx 0x66
6881 * @opcpuid clflushopt
6882 * @opgroup og_cachectl
6883 * @optest op1=1 ->
6884 */
6885FNIEMOP_DEF_1(iemOp_Grp15_clflushopt, uint8_t, bRm)
6886{
6887 IEMOP_MNEMONIC1(M_MEM, CLFLUSHOPT, clflushopt, Mb_RO, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
6888 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fClFlushOpt)
6889 return FNIEMOP_CALL_1(iemOp_InvalidWithRMAllNeeded, bRm);
6890
6891 IEM_MC_BEGIN(2, 0);
6892 IEM_MC_ARG(uint8_t, iEffSeg, 0);
6893 IEM_MC_ARG(RTGCPTR, GCPtrEff, 1);
6894 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
6895 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6896 IEM_MC_ASSIGN(iEffSeg, pVCpu->iem.s.iEffSeg);
6897 IEM_MC_CALL_CIMPL_2(iemCImpl_clflush_clflushopt, iEffSeg, GCPtrEff);
6898 IEM_MC_END();
6899 return VINF_SUCCESS;
6900}
6901
6902
6903/** Opcode 0x0f 0xae 11b/5. */
6904FNIEMOP_DEF_1(iemOp_Grp15_lfence, uint8_t, bRm)
6905{
6906 RT_NOREF_PV(bRm);
6907 IEMOP_MNEMONIC(lfence, "lfence");
6908 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6909 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2)
6910 return IEMOP_RAISE_INVALID_OPCODE();
6911
6912 IEM_MC_BEGIN(0, 0);
6913 if (IEM_GET_HOST_CPU_FEATURES(pVCpu)->fSse2)
6914 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_lfence);
6915 else
6916 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_alt_mem_fence);
6917 IEM_MC_ADVANCE_RIP();
6918 IEM_MC_END();
6919 return VINF_SUCCESS;
6920}
6921
6922
6923/** Opcode 0x0f 0xae 11b/6. */
6924FNIEMOP_DEF_1(iemOp_Grp15_mfence, uint8_t, bRm)
6925{
6926 RT_NOREF_PV(bRm);
6927 IEMOP_MNEMONIC(mfence, "mfence");
6928 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6929 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2)
6930 return IEMOP_RAISE_INVALID_OPCODE();
6931
6932 IEM_MC_BEGIN(0, 0);
6933 if (IEM_GET_HOST_CPU_FEATURES(pVCpu)->fSse2)
6934 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_mfence);
6935 else
6936 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_alt_mem_fence);
6937 IEM_MC_ADVANCE_RIP();
6938 IEM_MC_END();
6939 return VINF_SUCCESS;
6940}
6941
6942
6943/** Opcode 0x0f 0xae 11b/7. */
6944FNIEMOP_DEF_1(iemOp_Grp15_sfence, uint8_t, bRm)
6945{
6946 RT_NOREF_PV(bRm);
6947 IEMOP_MNEMONIC(sfence, "sfence");
6948 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
6949 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2)
6950 return IEMOP_RAISE_INVALID_OPCODE();
6951
6952 IEM_MC_BEGIN(0, 0);
6953 if (IEM_GET_HOST_CPU_FEATURES(pVCpu)->fSse2)
6954 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_sfence);
6955 else
6956 IEM_MC_CALL_VOID_AIMPL_0(iemAImpl_alt_mem_fence);
6957 IEM_MC_ADVANCE_RIP();
6958 IEM_MC_END();
6959 return VINF_SUCCESS;
6960}
6961
6962
6963/** Opcode 0xf3 0x0f 0xae 11b/0. */
6964FNIEMOP_UD_STUB_1(iemOp_Grp15_rdfsbase, uint8_t, bRm);
6965
6966/** Opcode 0xf3 0x0f 0xae 11b/1. */
6967FNIEMOP_UD_STUB_1(iemOp_Grp15_rdgsbase, uint8_t, bRm);
6968
6969/** Opcode 0xf3 0x0f 0xae 11b/2. */
6970FNIEMOP_UD_STUB_1(iemOp_Grp15_wrfsbase, uint8_t, bRm);
6971
6972/** Opcode 0xf3 0x0f 0xae 11b/3. */
6973FNIEMOP_UD_STUB_1(iemOp_Grp15_wrgsbase, uint8_t, bRm);
6974
6975
6976/**
6977 * Group 15 jump table for register variant.
6978 */
6979IEM_STATIC const PFNIEMOPRM g_apfnGroup15RegReg[] =
6980{ /* pfx: none, 066h, 0f3h, 0f2h */
6981 /* /0 */ iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_Grp15_rdfsbase, iemOp_InvalidWithRM,
6982 /* /1 */ iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_Grp15_rdgsbase, iemOp_InvalidWithRM,
6983 /* /2 */ iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_Grp15_wrfsbase, iemOp_InvalidWithRM,
6984 /* /3 */ iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_Grp15_wrgsbase, iemOp_InvalidWithRM,
6985 /* /4 */ IEMOP_X4(iemOp_InvalidWithRM),
6986 /* /5 */ iemOp_Grp15_lfence, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
6987 /* /6 */ iemOp_Grp15_mfence, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
6988 /* /7 */ iemOp_Grp15_sfence, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
6989};
6990AssertCompile(RT_ELEMENTS(g_apfnGroup15RegReg) == 8*4);
6991
6992
6993/**
6994 * Group 15 jump table for memory variant.
6995 */
6996IEM_STATIC const PFNIEMOPRM g_apfnGroup15MemReg[] =
6997{ /* pfx: none, 066h, 0f3h, 0f2h */
6998 /* /0 */ iemOp_Grp15_fxsave, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
6999 /* /1 */ iemOp_Grp15_fxrstor, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7000 /* /2 */ iemOp_Grp15_ldmxcsr, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7001 /* /3 */ iemOp_Grp15_stmxcsr, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7002 /* /4 */ iemOp_Grp15_xsave, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7003 /* /5 */ iemOp_Grp15_xrstor, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7004 /* /6 */ iemOp_Grp15_xsaveopt, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7005 /* /7 */ iemOp_Grp15_clflush, iemOp_Grp15_clflushopt, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
7006};
7007AssertCompile(RT_ELEMENTS(g_apfnGroup15MemReg) == 8*4);
7008
7009
7010/** Opcode 0x0f 0xae. */
7011FNIEMOP_DEF(iemOp_Grp15)
7012{
7013 IEMOP_HLP_MIN_586(); /* Not entirely accurate nor needed, but useful for debugging 286 code. */
7014 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7015 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7016 /* register, register */
7017 return FNIEMOP_CALL_1(g_apfnGroup15RegReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
7018 + pVCpu->iem.s.idxPrefix], bRm);
7019 /* memory, register */
7020 return FNIEMOP_CALL_1(g_apfnGroup15MemReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
7021 + pVCpu->iem.s.idxPrefix], bRm);
7022}
7023
7024
7025/** Opcode 0x0f 0xaf. */
7026FNIEMOP_DEF(iemOp_imul_Gv_Ev)
7027{
7028 IEMOP_MNEMONIC(imul_Gv_Ev, "imul Gv,Ev");
7029 IEMOP_HLP_MIN_386();
7030 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF);
7031 return FNIEMOP_CALL_1(iemOpHlpBinaryOperator_rv_rm, &g_iemAImpl_imul_two);
7032}
7033
7034
7035/** Opcode 0x0f 0xb0. */
7036FNIEMOP_DEF(iemOp_cmpxchg_Eb_Gb)
7037{
7038 IEMOP_MNEMONIC(cmpxchg_Eb_Gb, "cmpxchg Eb,Gb");
7039 IEMOP_HLP_MIN_486();
7040 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7041
7042 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7043 {
7044 IEMOP_HLP_DONE_DECODING();
7045 IEM_MC_BEGIN(4, 0);
7046 IEM_MC_ARG(uint8_t *, pu8Dst, 0);
7047 IEM_MC_ARG(uint8_t *, pu8Al, 1);
7048 IEM_MC_ARG(uint8_t, u8Src, 2);
7049 IEM_MC_ARG(uint32_t *, pEFlags, 3);
7050
7051 IEM_MC_FETCH_GREG_U8(u8Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7052 IEM_MC_REF_GREG_U8(pu8Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7053 IEM_MC_REF_GREG_U8(pu8Al, X86_GREG_xAX);
7054 IEM_MC_REF_EFLAGS(pEFlags);
7055 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7056 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u8, pu8Dst, pu8Al, u8Src, pEFlags);
7057 else
7058 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u8_locked, pu8Dst, pu8Al, u8Src, pEFlags);
7059
7060 IEM_MC_ADVANCE_RIP();
7061 IEM_MC_END();
7062 }
7063 else
7064 {
7065 IEM_MC_BEGIN(4, 3);
7066 IEM_MC_ARG(uint8_t *, pu8Dst, 0);
7067 IEM_MC_ARG(uint8_t *, pu8Al, 1);
7068 IEM_MC_ARG(uint8_t, u8Src, 2);
7069 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
7070 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7071 IEM_MC_LOCAL(uint8_t, u8Al);
7072
7073 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7074 IEMOP_HLP_DONE_DECODING();
7075 IEM_MC_MEM_MAP(pu8Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7076 IEM_MC_FETCH_GREG_U8(u8Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7077 IEM_MC_FETCH_GREG_U8(u8Al, X86_GREG_xAX);
7078 IEM_MC_FETCH_EFLAGS(EFlags);
7079 IEM_MC_REF_LOCAL(pu8Al, u8Al);
7080 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7081 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u8, pu8Dst, pu8Al, u8Src, pEFlags);
7082 else
7083 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u8_locked, pu8Dst, pu8Al, u8Src, pEFlags);
7084
7085 IEM_MC_MEM_COMMIT_AND_UNMAP(pu8Dst, IEM_ACCESS_DATA_RW);
7086 IEM_MC_COMMIT_EFLAGS(EFlags);
7087 IEM_MC_STORE_GREG_U8(X86_GREG_xAX, u8Al);
7088 IEM_MC_ADVANCE_RIP();
7089 IEM_MC_END();
7090 }
7091 return VINF_SUCCESS;
7092}
7093
7094/** Opcode 0x0f 0xb1. */
7095FNIEMOP_DEF(iemOp_cmpxchg_Ev_Gv)
7096{
7097 IEMOP_MNEMONIC(cmpxchg_Ev_Gv, "cmpxchg Ev,Gv");
7098 IEMOP_HLP_MIN_486();
7099 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7100
7101 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7102 {
7103 IEMOP_HLP_DONE_DECODING();
7104 switch (pVCpu->iem.s.enmEffOpSize)
7105 {
7106 case IEMMODE_16BIT:
7107 IEM_MC_BEGIN(4, 0);
7108 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
7109 IEM_MC_ARG(uint16_t *, pu16Ax, 1);
7110 IEM_MC_ARG(uint16_t, u16Src, 2);
7111 IEM_MC_ARG(uint32_t *, pEFlags, 3);
7112
7113 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7114 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7115 IEM_MC_REF_GREG_U16(pu16Ax, X86_GREG_xAX);
7116 IEM_MC_REF_EFLAGS(pEFlags);
7117 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7118 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u16, pu16Dst, pu16Ax, u16Src, pEFlags);
7119 else
7120 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u16_locked, pu16Dst, pu16Ax, u16Src, pEFlags);
7121
7122 IEM_MC_ADVANCE_RIP();
7123 IEM_MC_END();
7124 return VINF_SUCCESS;
7125
7126 case IEMMODE_32BIT:
7127 IEM_MC_BEGIN(4, 0);
7128 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
7129 IEM_MC_ARG(uint32_t *, pu32Eax, 1);
7130 IEM_MC_ARG(uint32_t, u32Src, 2);
7131 IEM_MC_ARG(uint32_t *, pEFlags, 3);
7132
7133 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7134 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7135 IEM_MC_REF_GREG_U32(pu32Eax, X86_GREG_xAX);
7136 IEM_MC_REF_EFLAGS(pEFlags);
7137 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7138 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u32, pu32Dst, pu32Eax, u32Src, pEFlags);
7139 else
7140 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u32_locked, pu32Dst, pu32Eax, u32Src, pEFlags);
7141
7142 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Eax);
7143 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
7144 IEM_MC_ADVANCE_RIP();
7145 IEM_MC_END();
7146 return VINF_SUCCESS;
7147
7148 case IEMMODE_64BIT:
7149 IEM_MC_BEGIN(4, 0);
7150 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
7151 IEM_MC_ARG(uint64_t *, pu64Rax, 1);
7152#ifdef RT_ARCH_X86
7153 IEM_MC_ARG(uint64_t *, pu64Src, 2);
7154#else
7155 IEM_MC_ARG(uint64_t, u64Src, 2);
7156#endif
7157 IEM_MC_ARG(uint32_t *, pEFlags, 3);
7158
7159 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7160 IEM_MC_REF_GREG_U64(pu64Rax, X86_GREG_xAX);
7161 IEM_MC_REF_EFLAGS(pEFlags);
7162#ifdef RT_ARCH_X86
7163 IEM_MC_REF_GREG_U64(pu64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7164 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7165 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64, pu64Dst, pu64Rax, pu64Src, pEFlags);
7166 else
7167 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64_locked, pu64Dst, pu64Rax, pu64Src, pEFlags);
7168#else
7169 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7170 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7171 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64, pu64Dst, pu64Rax, u64Src, pEFlags);
7172 else
7173 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64_locked, pu64Dst, pu64Rax, u64Src, pEFlags);
7174#endif
7175
7176 IEM_MC_ADVANCE_RIP();
7177 IEM_MC_END();
7178 return VINF_SUCCESS;
7179
7180 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7181 }
7182 }
7183 else
7184 {
7185 switch (pVCpu->iem.s.enmEffOpSize)
7186 {
7187 case IEMMODE_16BIT:
7188 IEM_MC_BEGIN(4, 3);
7189 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
7190 IEM_MC_ARG(uint16_t *, pu16Ax, 1);
7191 IEM_MC_ARG(uint16_t, u16Src, 2);
7192 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
7193 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7194 IEM_MC_LOCAL(uint16_t, u16Ax);
7195
7196 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7197 IEMOP_HLP_DONE_DECODING();
7198 IEM_MC_MEM_MAP(pu16Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7199 IEM_MC_FETCH_GREG_U16(u16Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7200 IEM_MC_FETCH_GREG_U16(u16Ax, X86_GREG_xAX);
7201 IEM_MC_FETCH_EFLAGS(EFlags);
7202 IEM_MC_REF_LOCAL(pu16Ax, u16Ax);
7203 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7204 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u16, pu16Dst, pu16Ax, u16Src, pEFlags);
7205 else
7206 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u16_locked, pu16Dst, pu16Ax, u16Src, pEFlags);
7207
7208 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, IEM_ACCESS_DATA_RW);
7209 IEM_MC_COMMIT_EFLAGS(EFlags);
7210 IEM_MC_STORE_GREG_U16(X86_GREG_xAX, u16Ax);
7211 IEM_MC_ADVANCE_RIP();
7212 IEM_MC_END();
7213 return VINF_SUCCESS;
7214
7215 case IEMMODE_32BIT:
7216 IEM_MC_BEGIN(4, 3);
7217 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
7218 IEM_MC_ARG(uint32_t *, pu32Eax, 1);
7219 IEM_MC_ARG(uint32_t, u32Src, 2);
7220 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
7221 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7222 IEM_MC_LOCAL(uint32_t, u32Eax);
7223
7224 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7225 IEMOP_HLP_DONE_DECODING();
7226 IEM_MC_MEM_MAP(pu32Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7227 IEM_MC_FETCH_GREG_U32(u32Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7228 IEM_MC_FETCH_GREG_U32(u32Eax, X86_GREG_xAX);
7229 IEM_MC_FETCH_EFLAGS(EFlags);
7230 IEM_MC_REF_LOCAL(pu32Eax, u32Eax);
7231 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7232 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u32, pu32Dst, pu32Eax, u32Src, pEFlags);
7233 else
7234 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u32_locked, pu32Dst, pu32Eax, u32Src, pEFlags);
7235
7236 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, IEM_ACCESS_DATA_RW);
7237 IEM_MC_COMMIT_EFLAGS(EFlags);
7238 IEM_MC_STORE_GREG_U32(X86_GREG_xAX, u32Eax);
7239 IEM_MC_ADVANCE_RIP();
7240 IEM_MC_END();
7241 return VINF_SUCCESS;
7242
7243 case IEMMODE_64BIT:
7244 IEM_MC_BEGIN(4, 3);
7245 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
7246 IEM_MC_ARG(uint64_t *, pu64Rax, 1);
7247#ifdef RT_ARCH_X86
7248 IEM_MC_ARG(uint64_t *, pu64Src, 2);
7249#else
7250 IEM_MC_ARG(uint64_t, u64Src, 2);
7251#endif
7252 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 3);
7253 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7254 IEM_MC_LOCAL(uint64_t, u64Rax);
7255
7256 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7257 IEMOP_HLP_DONE_DECODING();
7258 IEM_MC_MEM_MAP(pu64Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7259 IEM_MC_FETCH_GREG_U64(u64Rax, X86_GREG_xAX);
7260 IEM_MC_FETCH_EFLAGS(EFlags);
7261 IEM_MC_REF_LOCAL(pu64Rax, u64Rax);
7262#ifdef RT_ARCH_X86
7263 IEM_MC_REF_GREG_U64(pu64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7264 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7265 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64, pu64Dst, pu64Rax, pu64Src, pEFlags);
7266 else
7267 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64_locked, pu64Dst, pu64Rax, pu64Src, pEFlags);
7268#else
7269 IEM_MC_FETCH_GREG_U64(u64Src, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7270 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7271 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64, pu64Dst, pu64Rax, u64Src, pEFlags);
7272 else
7273 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg_u64_locked, pu64Dst, pu64Rax, u64Src, pEFlags);
7274#endif
7275
7276 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, IEM_ACCESS_DATA_RW);
7277 IEM_MC_COMMIT_EFLAGS(EFlags);
7278 IEM_MC_STORE_GREG_U64(X86_GREG_xAX, u64Rax);
7279 IEM_MC_ADVANCE_RIP();
7280 IEM_MC_END();
7281 return VINF_SUCCESS;
7282
7283 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7284 }
7285 }
7286}
7287
7288
7289FNIEMOP_DEF_2(iemOpCommonLoadSRegAndGreg, uint8_t, iSegReg, uint8_t, bRm)
7290{
7291 Assert((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT)); /* Caller checks this */
7292 uint8_t const iGReg = ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg;
7293
7294 switch (pVCpu->iem.s.enmEffOpSize)
7295 {
7296 case IEMMODE_16BIT:
7297 IEM_MC_BEGIN(5, 1);
7298 IEM_MC_ARG(uint16_t, uSel, 0);
7299 IEM_MC_ARG(uint16_t, offSeg, 1);
7300 IEM_MC_ARG_CONST(uint8_t, iSegRegArg,/*=*/iSegReg, 2);
7301 IEM_MC_ARG_CONST(uint8_t, iGRegArg, /*=*/iGReg, 3);
7302 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 4);
7303 IEM_MC_LOCAL(RTGCPTR, GCPtrEff);
7304 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
7305 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7306 IEM_MC_FETCH_MEM_U16(offSeg, pVCpu->iem.s.iEffSeg, GCPtrEff);
7307 IEM_MC_FETCH_MEM_U16_DISP(uSel, pVCpu->iem.s.iEffSeg, GCPtrEff, 2);
7308 IEM_MC_CALL_CIMPL_5(iemCImpl_load_SReg_Greg, uSel, offSeg, iSegRegArg, iGRegArg, enmEffOpSize);
7309 IEM_MC_END();
7310 return VINF_SUCCESS;
7311
7312 case IEMMODE_32BIT:
7313 IEM_MC_BEGIN(5, 1);
7314 IEM_MC_ARG(uint16_t, uSel, 0);
7315 IEM_MC_ARG(uint32_t, offSeg, 1);
7316 IEM_MC_ARG_CONST(uint8_t, iSegRegArg,/*=*/iSegReg, 2);
7317 IEM_MC_ARG_CONST(uint8_t, iGRegArg, /*=*/iGReg, 3);
7318 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 4);
7319 IEM_MC_LOCAL(RTGCPTR, GCPtrEff);
7320 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
7321 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7322 IEM_MC_FETCH_MEM_U32(offSeg, pVCpu->iem.s.iEffSeg, GCPtrEff);
7323 IEM_MC_FETCH_MEM_U16_DISP(uSel, pVCpu->iem.s.iEffSeg, GCPtrEff, 4);
7324 IEM_MC_CALL_CIMPL_5(iemCImpl_load_SReg_Greg, uSel, offSeg, iSegRegArg, iGRegArg, enmEffOpSize);
7325 IEM_MC_END();
7326 return VINF_SUCCESS;
7327
7328 case IEMMODE_64BIT:
7329 IEM_MC_BEGIN(5, 1);
7330 IEM_MC_ARG(uint16_t, uSel, 0);
7331 IEM_MC_ARG(uint64_t, offSeg, 1);
7332 IEM_MC_ARG_CONST(uint8_t, iSegRegArg,/*=*/iSegReg, 2);
7333 IEM_MC_ARG_CONST(uint8_t, iGRegArg, /*=*/iGReg, 3);
7334 IEM_MC_ARG_CONST(IEMMODE, enmEffOpSize,/*=*/pVCpu->iem.s.enmEffOpSize, 4);
7335 IEM_MC_LOCAL(RTGCPTR, GCPtrEff);
7336 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEff, bRm, 0);
7337 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7338 if (IEM_IS_GUEST_CPU_AMD(pVCpu)) /** @todo testcase: rev 3.15 of the amd manuals claims it only loads a 32-bit greg. */
7339 IEM_MC_FETCH_MEM_U32_SX_U64(offSeg, pVCpu->iem.s.iEffSeg, GCPtrEff);
7340 else
7341 IEM_MC_FETCH_MEM_U64(offSeg, pVCpu->iem.s.iEffSeg, GCPtrEff);
7342 IEM_MC_FETCH_MEM_U16_DISP(uSel, pVCpu->iem.s.iEffSeg, GCPtrEff, 8);
7343 IEM_MC_CALL_CIMPL_5(iemCImpl_load_SReg_Greg, uSel, offSeg, iSegRegArg, iGRegArg, enmEffOpSize);
7344 IEM_MC_END();
7345 return VINF_SUCCESS;
7346
7347 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7348 }
7349}
7350
7351
7352/** Opcode 0x0f 0xb2. */
7353FNIEMOP_DEF(iemOp_lss_Gv_Mp)
7354{
7355 IEMOP_MNEMONIC(lss_Gv_Mp, "lss Gv,Mp");
7356 IEMOP_HLP_MIN_386();
7357 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7358 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7359 return IEMOP_RAISE_INVALID_OPCODE();
7360 return FNIEMOP_CALL_2(iemOpCommonLoadSRegAndGreg, X86_SREG_SS, bRm);
7361}
7362
7363
7364/** Opcode 0x0f 0xb3. */
7365FNIEMOP_DEF(iemOp_btr_Ev_Gv)
7366{
7367 IEMOP_MNEMONIC(btr_Ev_Gv, "btr Ev,Gv");
7368 IEMOP_HLP_MIN_386();
7369 return FNIEMOP_CALL_1(iemOpCommonBit_Ev_Gv, &g_iemAImpl_btr);
7370}
7371
7372
7373/** Opcode 0x0f 0xb4. */
7374FNIEMOP_DEF(iemOp_lfs_Gv_Mp)
7375{
7376 IEMOP_MNEMONIC(lfs_Gv_Mp, "lfs Gv,Mp");
7377 IEMOP_HLP_MIN_386();
7378 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7379 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7380 return IEMOP_RAISE_INVALID_OPCODE();
7381 return FNIEMOP_CALL_2(iemOpCommonLoadSRegAndGreg, X86_SREG_FS, bRm);
7382}
7383
7384
7385/** Opcode 0x0f 0xb5. */
7386FNIEMOP_DEF(iemOp_lgs_Gv_Mp)
7387{
7388 IEMOP_MNEMONIC(lgs_Gv_Mp, "lgs Gv,Mp");
7389 IEMOP_HLP_MIN_386();
7390 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7391 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7392 return IEMOP_RAISE_INVALID_OPCODE();
7393 return FNIEMOP_CALL_2(iemOpCommonLoadSRegAndGreg, X86_SREG_GS, bRm);
7394}
7395
7396
7397/** Opcode 0x0f 0xb6. */
7398FNIEMOP_DEF(iemOp_movzx_Gv_Eb)
7399{
7400 IEMOP_MNEMONIC(movzx_Gv_Eb, "movzx Gv,Eb");
7401 IEMOP_HLP_MIN_386();
7402
7403 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7404
7405 /*
7406 * If rm is denoting a register, no more instruction bytes.
7407 */
7408 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7409 {
7410 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7411 switch (pVCpu->iem.s.enmEffOpSize)
7412 {
7413 case IEMMODE_16BIT:
7414 IEM_MC_BEGIN(0, 1);
7415 IEM_MC_LOCAL(uint16_t, u16Value);
7416 IEM_MC_FETCH_GREG_U8_ZX_U16(u16Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7417 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Value);
7418 IEM_MC_ADVANCE_RIP();
7419 IEM_MC_END();
7420 return VINF_SUCCESS;
7421
7422 case IEMMODE_32BIT:
7423 IEM_MC_BEGIN(0, 1);
7424 IEM_MC_LOCAL(uint32_t, u32Value);
7425 IEM_MC_FETCH_GREG_U8_ZX_U32(u32Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7426 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7427 IEM_MC_ADVANCE_RIP();
7428 IEM_MC_END();
7429 return VINF_SUCCESS;
7430
7431 case IEMMODE_64BIT:
7432 IEM_MC_BEGIN(0, 1);
7433 IEM_MC_LOCAL(uint64_t, u64Value);
7434 IEM_MC_FETCH_GREG_U8_ZX_U64(u64Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7435 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7436 IEM_MC_ADVANCE_RIP();
7437 IEM_MC_END();
7438 return VINF_SUCCESS;
7439
7440 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7441 }
7442 }
7443 else
7444 {
7445 /*
7446 * We're loading a register from memory.
7447 */
7448 switch (pVCpu->iem.s.enmEffOpSize)
7449 {
7450 case IEMMODE_16BIT:
7451 IEM_MC_BEGIN(0, 2);
7452 IEM_MC_LOCAL(uint16_t, u16Value);
7453 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7454 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7455 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7456 IEM_MC_FETCH_MEM_U8_ZX_U16(u16Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7457 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Value);
7458 IEM_MC_ADVANCE_RIP();
7459 IEM_MC_END();
7460 return VINF_SUCCESS;
7461
7462 case IEMMODE_32BIT:
7463 IEM_MC_BEGIN(0, 2);
7464 IEM_MC_LOCAL(uint32_t, u32Value);
7465 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7466 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7467 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7468 IEM_MC_FETCH_MEM_U8_ZX_U32(u32Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7469 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7470 IEM_MC_ADVANCE_RIP();
7471 IEM_MC_END();
7472 return VINF_SUCCESS;
7473
7474 case IEMMODE_64BIT:
7475 IEM_MC_BEGIN(0, 2);
7476 IEM_MC_LOCAL(uint64_t, u64Value);
7477 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7478 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7479 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7480 IEM_MC_FETCH_MEM_U8_ZX_U64(u64Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7481 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7482 IEM_MC_ADVANCE_RIP();
7483 IEM_MC_END();
7484 return VINF_SUCCESS;
7485
7486 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7487 }
7488 }
7489}
7490
7491
7492/** Opcode 0x0f 0xb7. */
7493FNIEMOP_DEF(iemOp_movzx_Gv_Ew)
7494{
7495 IEMOP_MNEMONIC(movzx_Gv_Ew, "movzx Gv,Ew");
7496 IEMOP_HLP_MIN_386();
7497
7498 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7499
7500 /** @todo Not entirely sure how the operand size prefix is handled here,
7501 * assuming that it will be ignored. Would be nice to have a few
7502 * test for this. */
7503 /*
7504 * If rm is denoting a register, no more instruction bytes.
7505 */
7506 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7507 {
7508 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7509 if (pVCpu->iem.s.enmEffOpSize != IEMMODE_64BIT)
7510 {
7511 IEM_MC_BEGIN(0, 1);
7512 IEM_MC_LOCAL(uint32_t, u32Value);
7513 IEM_MC_FETCH_GREG_U16_ZX_U32(u32Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7514 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7515 IEM_MC_ADVANCE_RIP();
7516 IEM_MC_END();
7517 }
7518 else
7519 {
7520 IEM_MC_BEGIN(0, 1);
7521 IEM_MC_LOCAL(uint64_t, u64Value);
7522 IEM_MC_FETCH_GREG_U16_ZX_U64(u64Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7523 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7524 IEM_MC_ADVANCE_RIP();
7525 IEM_MC_END();
7526 }
7527 }
7528 else
7529 {
7530 /*
7531 * We're loading a register from memory.
7532 */
7533 if (pVCpu->iem.s.enmEffOpSize != IEMMODE_64BIT)
7534 {
7535 IEM_MC_BEGIN(0, 2);
7536 IEM_MC_LOCAL(uint32_t, u32Value);
7537 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7538 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7539 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7540 IEM_MC_FETCH_MEM_U16_ZX_U32(u32Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7541 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7542 IEM_MC_ADVANCE_RIP();
7543 IEM_MC_END();
7544 }
7545 else
7546 {
7547 IEM_MC_BEGIN(0, 2);
7548 IEM_MC_LOCAL(uint64_t, u64Value);
7549 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7550 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7551 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7552 IEM_MC_FETCH_MEM_U16_ZX_U64(u64Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7553 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7554 IEM_MC_ADVANCE_RIP();
7555 IEM_MC_END();
7556 }
7557 }
7558 return VINF_SUCCESS;
7559}
7560
7561
7562/** Opcode 0x0f 0xb8 - JMPE (reserved for emulator on IPF) */
7563FNIEMOP_UD_STUB(iemOp_jmpe);
7564/** Opcode 0xf3 0x0f 0xb8 - POPCNT Gv, Ev */
7565FNIEMOP_STUB(iemOp_popcnt_Gv_Ev);
7566
7567
7568/**
7569 * @opcode 0xb9
7570 * @opinvalid intel-modrm
7571 * @optest ->
7572 */
7573FNIEMOP_DEF(iemOp_Grp10)
7574{
7575 /*
7576 * AMD does not decode beyond the 0xb9 whereas intel does the modr/m bit
7577 * too. See bs3-cpu-decoder-1.c32. So, we can forward to iemOp_InvalidNeedRM.
7578 */
7579 Log(("iemOp_Grp10 aka UD1 -> #UD\n"));
7580 IEMOP_MNEMONIC2EX(ud1, "ud1", RM, UD1, ud1, Gb, Eb, DISOPTYPE_INVALID, IEMOPHINT_IGNORES_OP_SIZES); /* just picked Gb,Eb here. */
7581 return FNIEMOP_CALL(iemOp_InvalidNeedRM);
7582}
7583
7584
7585/** Opcode 0x0f 0xba. */
7586FNIEMOP_DEF(iemOp_Grp8)
7587{
7588 IEMOP_HLP_MIN_386();
7589 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7590 PCIEMOPBINSIZES pImpl;
7591 switch ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK)
7592 {
7593 case 0: case 1: case 2: case 3:
7594 /* Both AMD and Intel want full modr/m decoding and imm8. */
7595 return FNIEMOP_CALL_1(iemOp_InvalidWithRMAllNeedImm8, bRm);
7596 case 4: pImpl = &g_iemAImpl_bt; IEMOP_MNEMONIC(bt_Ev_Ib, "bt Ev,Ib"); break;
7597 case 5: pImpl = &g_iemAImpl_bts; IEMOP_MNEMONIC(bts_Ev_Ib, "bts Ev,Ib"); break;
7598 case 6: pImpl = &g_iemAImpl_btr; IEMOP_MNEMONIC(btr_Ev_Ib, "btr Ev,Ib"); break;
7599 case 7: pImpl = &g_iemAImpl_btc; IEMOP_MNEMONIC(btc_Ev_Ib, "btc Ev,Ib"); break;
7600 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7601 }
7602 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_OF | X86_EFL_SF | X86_EFL_ZF | X86_EFL_AF | X86_EFL_PF);
7603
7604 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7605 {
7606 /* register destination. */
7607 uint8_t u8Bit; IEM_OPCODE_GET_NEXT_U8(&u8Bit);
7608 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7609
7610 switch (pVCpu->iem.s.enmEffOpSize)
7611 {
7612 case IEMMODE_16BIT:
7613 IEM_MC_BEGIN(3, 0);
7614 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
7615 IEM_MC_ARG_CONST(uint16_t, u16Src, /*=*/ u8Bit & 0x0f, 1);
7616 IEM_MC_ARG(uint32_t *, pEFlags, 2);
7617
7618 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7619 IEM_MC_REF_EFLAGS(pEFlags);
7620 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU16, pu16Dst, u16Src, pEFlags);
7621
7622 IEM_MC_ADVANCE_RIP();
7623 IEM_MC_END();
7624 return VINF_SUCCESS;
7625
7626 case IEMMODE_32BIT:
7627 IEM_MC_BEGIN(3, 0);
7628 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
7629 IEM_MC_ARG_CONST(uint32_t, u32Src, /*=*/ u8Bit & 0x1f, 1);
7630 IEM_MC_ARG(uint32_t *, pEFlags, 2);
7631
7632 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7633 IEM_MC_REF_EFLAGS(pEFlags);
7634 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU32, pu32Dst, u32Src, pEFlags);
7635
7636 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
7637 IEM_MC_ADVANCE_RIP();
7638 IEM_MC_END();
7639 return VINF_SUCCESS;
7640
7641 case IEMMODE_64BIT:
7642 IEM_MC_BEGIN(3, 0);
7643 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
7644 IEM_MC_ARG_CONST(uint64_t, u64Src, /*=*/ u8Bit & 0x3f, 1);
7645 IEM_MC_ARG(uint32_t *, pEFlags, 2);
7646
7647 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7648 IEM_MC_REF_EFLAGS(pEFlags);
7649 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU64, pu64Dst, u64Src, pEFlags);
7650
7651 IEM_MC_ADVANCE_RIP();
7652 IEM_MC_END();
7653 return VINF_SUCCESS;
7654
7655 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7656 }
7657 }
7658 else
7659 {
7660 /* memory destination. */
7661
7662 uint32_t fAccess;
7663 if (pImpl->pfnLockedU16)
7664 fAccess = IEM_ACCESS_DATA_RW;
7665 else /* BT */
7666 fAccess = IEM_ACCESS_DATA_R;
7667
7668 /** @todo test negative bit offsets! */
7669 switch (pVCpu->iem.s.enmEffOpSize)
7670 {
7671 case IEMMODE_16BIT:
7672 IEM_MC_BEGIN(3, 1);
7673 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
7674 IEM_MC_ARG(uint16_t, u16Src, 1);
7675 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
7676 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7677
7678 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
7679 uint8_t u8Bit; IEM_OPCODE_GET_NEXT_U8(&u8Bit);
7680 IEM_MC_ASSIGN(u16Src, u8Bit & 0x0f);
7681 if (pImpl->pfnLockedU16)
7682 IEMOP_HLP_DONE_DECODING();
7683 else
7684 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7685 IEM_MC_FETCH_EFLAGS(EFlags);
7686 IEM_MC_MEM_MAP(pu16Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7687 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7688 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU16, pu16Dst, u16Src, pEFlags);
7689 else
7690 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU16, pu16Dst, u16Src, pEFlags);
7691 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, fAccess);
7692
7693 IEM_MC_COMMIT_EFLAGS(EFlags);
7694 IEM_MC_ADVANCE_RIP();
7695 IEM_MC_END();
7696 return VINF_SUCCESS;
7697
7698 case IEMMODE_32BIT:
7699 IEM_MC_BEGIN(3, 1);
7700 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
7701 IEM_MC_ARG(uint32_t, u32Src, 1);
7702 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
7703 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7704
7705 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
7706 uint8_t u8Bit; IEM_OPCODE_GET_NEXT_U8(&u8Bit);
7707 IEM_MC_ASSIGN(u32Src, u8Bit & 0x1f);
7708 if (pImpl->pfnLockedU16)
7709 IEMOP_HLP_DONE_DECODING();
7710 else
7711 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7712 IEM_MC_FETCH_EFLAGS(EFlags);
7713 IEM_MC_MEM_MAP(pu32Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7714 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7715 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU32, pu32Dst, u32Src, pEFlags);
7716 else
7717 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU32, pu32Dst, u32Src, pEFlags);
7718 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, fAccess);
7719
7720 IEM_MC_COMMIT_EFLAGS(EFlags);
7721 IEM_MC_ADVANCE_RIP();
7722 IEM_MC_END();
7723 return VINF_SUCCESS;
7724
7725 case IEMMODE_64BIT:
7726 IEM_MC_BEGIN(3, 1);
7727 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
7728 IEM_MC_ARG(uint64_t, u64Src, 1);
7729 IEM_MC_ARG_LOCAL_EFLAGS( pEFlags, EFlags, 2);
7730 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7731
7732 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 1);
7733 uint8_t u8Bit; IEM_OPCODE_GET_NEXT_U8(&u8Bit);
7734 IEM_MC_ASSIGN(u64Src, u8Bit & 0x3f);
7735 if (pImpl->pfnLockedU16)
7736 IEMOP_HLP_DONE_DECODING();
7737 else
7738 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7739 IEM_MC_FETCH_EFLAGS(EFlags);
7740 IEM_MC_MEM_MAP(pu64Dst, fAccess, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0);
7741 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
7742 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnNormalU64, pu64Dst, u64Src, pEFlags);
7743 else
7744 IEM_MC_CALL_VOID_AIMPL_3(pImpl->pfnLockedU64, pu64Dst, u64Src, pEFlags);
7745 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, fAccess);
7746
7747 IEM_MC_COMMIT_EFLAGS(EFlags);
7748 IEM_MC_ADVANCE_RIP();
7749 IEM_MC_END();
7750 return VINF_SUCCESS;
7751
7752 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7753 }
7754 }
7755}
7756
7757
7758/** Opcode 0x0f 0xbb. */
7759FNIEMOP_DEF(iemOp_btc_Ev_Gv)
7760{
7761 IEMOP_MNEMONIC(btc_Ev_Gv, "btc Ev,Gv");
7762 IEMOP_HLP_MIN_386();
7763 return FNIEMOP_CALL_1(iemOpCommonBit_Ev_Gv, &g_iemAImpl_btc);
7764}
7765
7766
7767/** Opcode 0x0f 0xbc. */
7768FNIEMOP_DEF(iemOp_bsf_Gv_Ev)
7769{
7770 IEMOP_MNEMONIC(bsf_Gv_Ev, "bsf Gv,Ev");
7771 IEMOP_HLP_MIN_386();
7772 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_OF | X86_EFL_SF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF);
7773 return FNIEMOP_CALL_1(iemOpHlpBinaryOperator_rv_rm, &g_iemAImpl_bsf);
7774}
7775
7776
7777/** Opcode 0xf3 0x0f 0xbc - TZCNT Gv, Ev */
7778FNIEMOP_STUB(iemOp_tzcnt_Gv_Ev);
7779
7780
7781/** Opcode 0x0f 0xbd. */
7782FNIEMOP_DEF(iemOp_bsr_Gv_Ev)
7783{
7784 IEMOP_MNEMONIC(bsr_Gv_Ev, "bsr Gv,Ev");
7785 IEMOP_HLP_MIN_386();
7786 IEMOP_VERIFICATION_UNDEFINED_EFLAGS(X86_EFL_OF | X86_EFL_SF | X86_EFL_AF | X86_EFL_PF | X86_EFL_CF);
7787 return FNIEMOP_CALL_1(iemOpHlpBinaryOperator_rv_rm, &g_iemAImpl_bsr);
7788}
7789
7790
7791/** Opcode 0xf3 0x0f 0xbd - LZCNT Gv, Ev */
7792FNIEMOP_STUB(iemOp_lzcnt_Gv_Ev);
7793
7794
7795/** Opcode 0x0f 0xbe. */
7796FNIEMOP_DEF(iemOp_movsx_Gv_Eb)
7797{
7798 IEMOP_MNEMONIC(movsx_Gv_Eb, "movsx Gv,Eb");
7799 IEMOP_HLP_MIN_386();
7800
7801 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7802
7803 /*
7804 * If rm is denoting a register, no more instruction bytes.
7805 */
7806 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7807 {
7808 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7809 switch (pVCpu->iem.s.enmEffOpSize)
7810 {
7811 case IEMMODE_16BIT:
7812 IEM_MC_BEGIN(0, 1);
7813 IEM_MC_LOCAL(uint16_t, u16Value);
7814 IEM_MC_FETCH_GREG_U8_SX_U16(u16Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7815 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Value);
7816 IEM_MC_ADVANCE_RIP();
7817 IEM_MC_END();
7818 return VINF_SUCCESS;
7819
7820 case IEMMODE_32BIT:
7821 IEM_MC_BEGIN(0, 1);
7822 IEM_MC_LOCAL(uint32_t, u32Value);
7823 IEM_MC_FETCH_GREG_U8_SX_U32(u32Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7824 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7825 IEM_MC_ADVANCE_RIP();
7826 IEM_MC_END();
7827 return VINF_SUCCESS;
7828
7829 case IEMMODE_64BIT:
7830 IEM_MC_BEGIN(0, 1);
7831 IEM_MC_LOCAL(uint64_t, u64Value);
7832 IEM_MC_FETCH_GREG_U8_SX_U64(u64Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7833 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7834 IEM_MC_ADVANCE_RIP();
7835 IEM_MC_END();
7836 return VINF_SUCCESS;
7837
7838 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7839 }
7840 }
7841 else
7842 {
7843 /*
7844 * We're loading a register from memory.
7845 */
7846 switch (pVCpu->iem.s.enmEffOpSize)
7847 {
7848 case IEMMODE_16BIT:
7849 IEM_MC_BEGIN(0, 2);
7850 IEM_MC_LOCAL(uint16_t, u16Value);
7851 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7852 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7853 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7854 IEM_MC_FETCH_MEM_U8_SX_U16(u16Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7855 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16Value);
7856 IEM_MC_ADVANCE_RIP();
7857 IEM_MC_END();
7858 return VINF_SUCCESS;
7859
7860 case IEMMODE_32BIT:
7861 IEM_MC_BEGIN(0, 2);
7862 IEM_MC_LOCAL(uint32_t, u32Value);
7863 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7864 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7865 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7866 IEM_MC_FETCH_MEM_U8_SX_U32(u32Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7867 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7868 IEM_MC_ADVANCE_RIP();
7869 IEM_MC_END();
7870 return VINF_SUCCESS;
7871
7872 case IEMMODE_64BIT:
7873 IEM_MC_BEGIN(0, 2);
7874 IEM_MC_LOCAL(uint64_t, u64Value);
7875 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7876 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7877 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7878 IEM_MC_FETCH_MEM_U8_SX_U64(u64Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7879 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7880 IEM_MC_ADVANCE_RIP();
7881 IEM_MC_END();
7882 return VINF_SUCCESS;
7883
7884 IEM_NOT_REACHED_DEFAULT_CASE_RET();
7885 }
7886 }
7887}
7888
7889
7890/** Opcode 0x0f 0xbf. */
7891FNIEMOP_DEF(iemOp_movsx_Gv_Ew)
7892{
7893 IEMOP_MNEMONIC(movsx_Gv_Ew, "movsx Gv,Ew");
7894 IEMOP_HLP_MIN_386();
7895
7896 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7897
7898 /** @todo Not entirely sure how the operand size prefix is handled here,
7899 * assuming that it will be ignored. Would be nice to have a few
7900 * test for this. */
7901 /*
7902 * If rm is denoting a register, no more instruction bytes.
7903 */
7904 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7905 {
7906 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7907 if (pVCpu->iem.s.enmEffOpSize != IEMMODE_64BIT)
7908 {
7909 IEM_MC_BEGIN(0, 1);
7910 IEM_MC_LOCAL(uint32_t, u32Value);
7911 IEM_MC_FETCH_GREG_U16_SX_U32(u32Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7912 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7913 IEM_MC_ADVANCE_RIP();
7914 IEM_MC_END();
7915 }
7916 else
7917 {
7918 IEM_MC_BEGIN(0, 1);
7919 IEM_MC_LOCAL(uint64_t, u64Value);
7920 IEM_MC_FETCH_GREG_U16_SX_U64(u64Value, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7921 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7922 IEM_MC_ADVANCE_RIP();
7923 IEM_MC_END();
7924 }
7925 }
7926 else
7927 {
7928 /*
7929 * We're loading a register from memory.
7930 */
7931 if (pVCpu->iem.s.enmEffOpSize != IEMMODE_64BIT)
7932 {
7933 IEM_MC_BEGIN(0, 2);
7934 IEM_MC_LOCAL(uint32_t, u32Value);
7935 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7936 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7937 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7938 IEM_MC_FETCH_MEM_U16_SX_U32(u32Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7939 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32Value);
7940 IEM_MC_ADVANCE_RIP();
7941 IEM_MC_END();
7942 }
7943 else
7944 {
7945 IEM_MC_BEGIN(0, 2);
7946 IEM_MC_LOCAL(uint64_t, u64Value);
7947 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7948 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
7949 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7950 IEM_MC_FETCH_MEM_U16_SX_U64(u64Value, pVCpu->iem.s.iEffSeg, GCPtrEffDst);
7951 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64Value);
7952 IEM_MC_ADVANCE_RIP();
7953 IEM_MC_END();
7954 }
7955 }
7956 return VINF_SUCCESS;
7957}
7958
7959
7960/** Opcode 0x0f 0xc0. */
7961FNIEMOP_DEF(iemOp_xadd_Eb_Gb)
7962{
7963 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
7964 IEMOP_HLP_MIN_486();
7965 IEMOP_MNEMONIC(xadd_Eb_Gb, "xadd Eb,Gb");
7966
7967 /*
7968 * If rm is denoting a register, no more instruction bytes.
7969 */
7970 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
7971 {
7972 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
7973
7974 IEM_MC_BEGIN(3, 0);
7975 IEM_MC_ARG(uint8_t *, pu8Dst, 0);
7976 IEM_MC_ARG(uint8_t *, pu8Reg, 1);
7977 IEM_MC_ARG(uint32_t *, pEFlags, 2);
7978
7979 IEM_MC_REF_GREG_U8(pu8Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
7980 IEM_MC_REF_GREG_U8(pu8Reg, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
7981 IEM_MC_REF_EFLAGS(pEFlags);
7982 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u8, pu8Dst, pu8Reg, pEFlags);
7983
7984 IEM_MC_ADVANCE_RIP();
7985 IEM_MC_END();
7986 }
7987 else
7988 {
7989 /*
7990 * We're accessing memory.
7991 */
7992 IEM_MC_BEGIN(3, 3);
7993 IEM_MC_ARG(uint8_t *, pu8Dst, 0);
7994 IEM_MC_ARG(uint8_t *, pu8Reg, 1);
7995 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 2);
7996 IEM_MC_LOCAL(uint8_t, u8RegCopy);
7997 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
7998
7999 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8000 IEM_MC_MEM_MAP(pu8Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8001 IEM_MC_FETCH_GREG_U8(u8RegCopy, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8002 IEM_MC_REF_LOCAL(pu8Reg, u8RegCopy);
8003 IEM_MC_FETCH_EFLAGS(EFlags);
8004 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8005 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u8, pu8Dst, pu8Reg, pEFlags);
8006 else
8007 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u8_locked, pu8Dst, pu8Reg, pEFlags);
8008
8009 IEM_MC_MEM_COMMIT_AND_UNMAP(pu8Dst, IEM_ACCESS_DATA_RW);
8010 IEM_MC_COMMIT_EFLAGS(EFlags);
8011 IEM_MC_STORE_GREG_U8(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u8RegCopy);
8012 IEM_MC_ADVANCE_RIP();
8013 IEM_MC_END();
8014 return VINF_SUCCESS;
8015 }
8016 return VINF_SUCCESS;
8017}
8018
8019
8020/** Opcode 0x0f 0xc1. */
8021FNIEMOP_DEF(iemOp_xadd_Ev_Gv)
8022{
8023 IEMOP_MNEMONIC(xadd_Ev_Gv, "xadd Ev,Gv");
8024 IEMOP_HLP_MIN_486();
8025 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8026
8027 /*
8028 * If rm is denoting a register, no more instruction bytes.
8029 */
8030 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
8031 {
8032 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8033
8034 switch (pVCpu->iem.s.enmEffOpSize)
8035 {
8036 case IEMMODE_16BIT:
8037 IEM_MC_BEGIN(3, 0);
8038 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
8039 IEM_MC_ARG(uint16_t *, pu16Reg, 1);
8040 IEM_MC_ARG(uint32_t *, pEFlags, 2);
8041
8042 IEM_MC_REF_GREG_U16(pu16Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
8043 IEM_MC_REF_GREG_U16(pu16Reg, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8044 IEM_MC_REF_EFLAGS(pEFlags);
8045 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u16, pu16Dst, pu16Reg, pEFlags);
8046
8047 IEM_MC_ADVANCE_RIP();
8048 IEM_MC_END();
8049 return VINF_SUCCESS;
8050
8051 case IEMMODE_32BIT:
8052 IEM_MC_BEGIN(3, 0);
8053 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
8054 IEM_MC_ARG(uint32_t *, pu32Reg, 1);
8055 IEM_MC_ARG(uint32_t *, pEFlags, 2);
8056
8057 IEM_MC_REF_GREG_U32(pu32Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
8058 IEM_MC_REF_GREG_U32(pu32Reg, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8059 IEM_MC_REF_EFLAGS(pEFlags);
8060 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u32, pu32Dst, pu32Reg, pEFlags);
8061
8062 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
8063 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Reg);
8064 IEM_MC_ADVANCE_RIP();
8065 IEM_MC_END();
8066 return VINF_SUCCESS;
8067
8068 case IEMMODE_64BIT:
8069 IEM_MC_BEGIN(3, 0);
8070 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
8071 IEM_MC_ARG(uint64_t *, pu64Reg, 1);
8072 IEM_MC_ARG(uint32_t *, pEFlags, 2);
8073
8074 IEM_MC_REF_GREG_U64(pu64Dst, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
8075 IEM_MC_REF_GREG_U64(pu64Reg, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8076 IEM_MC_REF_EFLAGS(pEFlags);
8077 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u64, pu64Dst, pu64Reg, pEFlags);
8078
8079 IEM_MC_ADVANCE_RIP();
8080 IEM_MC_END();
8081 return VINF_SUCCESS;
8082
8083 IEM_NOT_REACHED_DEFAULT_CASE_RET();
8084 }
8085 }
8086 else
8087 {
8088 /*
8089 * We're accessing memory.
8090 */
8091 switch (pVCpu->iem.s.enmEffOpSize)
8092 {
8093 case IEMMODE_16BIT:
8094 IEM_MC_BEGIN(3, 3);
8095 IEM_MC_ARG(uint16_t *, pu16Dst, 0);
8096 IEM_MC_ARG(uint16_t *, pu16Reg, 1);
8097 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 2);
8098 IEM_MC_LOCAL(uint16_t, u16RegCopy);
8099 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8100
8101 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8102 IEM_MC_MEM_MAP(pu16Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8103 IEM_MC_FETCH_GREG_U16(u16RegCopy, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8104 IEM_MC_REF_LOCAL(pu16Reg, u16RegCopy);
8105 IEM_MC_FETCH_EFLAGS(EFlags);
8106 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8107 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u16, pu16Dst, pu16Reg, pEFlags);
8108 else
8109 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u16_locked, pu16Dst, pu16Reg, pEFlags);
8110
8111 IEM_MC_MEM_COMMIT_AND_UNMAP(pu16Dst, IEM_ACCESS_DATA_RW);
8112 IEM_MC_COMMIT_EFLAGS(EFlags);
8113 IEM_MC_STORE_GREG_U16(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u16RegCopy);
8114 IEM_MC_ADVANCE_RIP();
8115 IEM_MC_END();
8116 return VINF_SUCCESS;
8117
8118 case IEMMODE_32BIT:
8119 IEM_MC_BEGIN(3, 3);
8120 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
8121 IEM_MC_ARG(uint32_t *, pu32Reg, 1);
8122 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 2);
8123 IEM_MC_LOCAL(uint32_t, u32RegCopy);
8124 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8125
8126 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8127 IEM_MC_MEM_MAP(pu32Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8128 IEM_MC_FETCH_GREG_U32(u32RegCopy, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8129 IEM_MC_REF_LOCAL(pu32Reg, u32RegCopy);
8130 IEM_MC_FETCH_EFLAGS(EFlags);
8131 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8132 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u32, pu32Dst, pu32Reg, pEFlags);
8133 else
8134 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u32_locked, pu32Dst, pu32Reg, pEFlags);
8135
8136 IEM_MC_MEM_COMMIT_AND_UNMAP(pu32Dst, IEM_ACCESS_DATA_RW);
8137 IEM_MC_COMMIT_EFLAGS(EFlags);
8138 IEM_MC_STORE_GREG_U32(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u32RegCopy);
8139 IEM_MC_ADVANCE_RIP();
8140 IEM_MC_END();
8141 return VINF_SUCCESS;
8142
8143 case IEMMODE_64BIT:
8144 IEM_MC_BEGIN(3, 3);
8145 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
8146 IEM_MC_ARG(uint64_t *, pu64Reg, 1);
8147 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 2);
8148 IEM_MC_LOCAL(uint64_t, u64RegCopy);
8149 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8150
8151 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8152 IEM_MC_MEM_MAP(pu64Dst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8153 IEM_MC_FETCH_GREG_U64(u64RegCopy, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8154 IEM_MC_REF_LOCAL(pu64Reg, u64RegCopy);
8155 IEM_MC_FETCH_EFLAGS(EFlags);
8156 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8157 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u64, pu64Dst, pu64Reg, pEFlags);
8158 else
8159 IEM_MC_CALL_VOID_AIMPL_3(iemAImpl_xadd_u64_locked, pu64Dst, pu64Reg, pEFlags);
8160
8161 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64Dst, IEM_ACCESS_DATA_RW);
8162 IEM_MC_COMMIT_EFLAGS(EFlags);
8163 IEM_MC_STORE_GREG_U64(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, u64RegCopy);
8164 IEM_MC_ADVANCE_RIP();
8165 IEM_MC_END();
8166 return VINF_SUCCESS;
8167
8168 IEM_NOT_REACHED_DEFAULT_CASE_RET();
8169 }
8170 }
8171}
8172
8173
8174/** Opcode 0x0f 0xc2 - cmpps Vps,Wps,Ib */
8175FNIEMOP_STUB(iemOp_cmpps_Vps_Wps_Ib);
8176/** Opcode 0x66 0x0f 0xc2 - cmppd Vpd,Wpd,Ib */
8177FNIEMOP_STUB(iemOp_cmppd_Vpd_Wpd_Ib);
8178/** Opcode 0xf3 0x0f 0xc2 - cmpss Vss,Wss,Ib */
8179FNIEMOP_STUB(iemOp_cmpss_Vss_Wss_Ib);
8180/** Opcode 0xf2 0x0f 0xc2 - cmpsd Vsd,Wsd,Ib */
8181FNIEMOP_STUB(iemOp_cmpsd_Vsd_Wsd_Ib);
8182
8183
8184/** Opcode 0x0f 0xc3. */
8185FNIEMOP_DEF(iemOp_movnti_My_Gy)
8186{
8187 IEMOP_MNEMONIC(movnti_My_Gy, "movnti My,Gy");
8188
8189 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8190
8191 /* Only the register -> memory form makes sense, assuming #UD for the other form. */
8192 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
8193 {
8194 switch (pVCpu->iem.s.enmEffOpSize)
8195 {
8196 case IEMMODE_32BIT:
8197 IEM_MC_BEGIN(0, 2);
8198 IEM_MC_LOCAL(uint32_t, u32Value);
8199 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8200
8201 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8202 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8203 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2)
8204 return IEMOP_RAISE_INVALID_OPCODE();
8205
8206 IEM_MC_FETCH_GREG_U32(u32Value, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8207 IEM_MC_STORE_MEM_U32(pVCpu->iem.s.iEffSeg, GCPtrEffDst, u32Value);
8208 IEM_MC_ADVANCE_RIP();
8209 IEM_MC_END();
8210 break;
8211
8212 case IEMMODE_64BIT:
8213 IEM_MC_BEGIN(0, 2);
8214 IEM_MC_LOCAL(uint64_t, u64Value);
8215 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8216
8217 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8218 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8219 if (!IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fSse2)
8220 return IEMOP_RAISE_INVALID_OPCODE();
8221
8222 IEM_MC_FETCH_GREG_U64(u64Value, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8223 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffDst, u64Value);
8224 IEM_MC_ADVANCE_RIP();
8225 IEM_MC_END();
8226 break;
8227
8228 case IEMMODE_16BIT:
8229 /** @todo check this form. */
8230 return IEMOP_RAISE_INVALID_OPCODE();
8231 }
8232 }
8233 else
8234 return IEMOP_RAISE_INVALID_OPCODE();
8235 return VINF_SUCCESS;
8236}
8237/* Opcode 0x66 0x0f 0xc3 - invalid */
8238/* Opcode 0xf3 0x0f 0xc3 - invalid */
8239/* Opcode 0xf2 0x0f 0xc3 - invalid */
8240
8241/** Opcode 0x0f 0xc4 - pinsrw Pq, Ry/Mw,Ib */
8242FNIEMOP_STUB(iemOp_pinsrw_Pq_RyMw_Ib);
8243/** Opcode 0x66 0x0f 0xc4 - pinsrw Vdq, Ry/Mw,Ib */
8244FNIEMOP_STUB(iemOp_pinsrw_Vdq_RyMw_Ib);
8245/* Opcode 0xf3 0x0f 0xc4 - invalid */
8246/* Opcode 0xf2 0x0f 0xc4 - invalid */
8247
8248/** Opcode 0x0f 0xc5 - pextrw Gd, Nq, Ib */
8249FNIEMOP_STUB(iemOp_pextrw_Gd_Nq_Ib);
8250/** Opcode 0x66 0x0f 0xc5 - pextrw Gd, Udq, Ib */
8251FNIEMOP_STUB(iemOp_pextrw_Gd_Udq_Ib);
8252/* Opcode 0xf3 0x0f 0xc5 - invalid */
8253/* Opcode 0xf2 0x0f 0xc5 - invalid */
8254
8255/** Opcode 0x0f 0xc6 - shufps Vps, Wps, Ib */
8256FNIEMOP_STUB(iemOp_shufps_Vps_Wps_Ib);
8257/** Opcode 0x66 0x0f 0xc6 - shufpd Vpd, Wpd, Ib */
8258FNIEMOP_STUB(iemOp_shufpd_Vpd_Wpd_Ib);
8259/* Opcode 0xf3 0x0f 0xc6 - invalid */
8260/* Opcode 0xf2 0x0f 0xc6 - invalid */
8261
8262
8263/** Opcode 0x0f 0xc7 !11/1. */
8264FNIEMOP_DEF_1(iemOp_Grp9_cmpxchg8b_Mq, uint8_t, bRm)
8265{
8266 IEMOP_MNEMONIC(cmpxchg8b, "cmpxchg8b Mq");
8267
8268 IEM_MC_BEGIN(4, 3);
8269 IEM_MC_ARG(uint64_t *, pu64MemDst, 0);
8270 IEM_MC_ARG(PRTUINT64U, pu64EaxEdx, 1);
8271 IEM_MC_ARG(PRTUINT64U, pu64EbxEcx, 2);
8272 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 3);
8273 IEM_MC_LOCAL(RTUINT64U, u64EaxEdx);
8274 IEM_MC_LOCAL(RTUINT64U, u64EbxEcx);
8275 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8276
8277 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8278 IEMOP_HLP_DONE_DECODING();
8279 IEM_MC_MEM_MAP(pu64MemDst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8280
8281 IEM_MC_FETCH_GREG_U32(u64EaxEdx.s.Lo, X86_GREG_xAX);
8282 IEM_MC_FETCH_GREG_U32(u64EaxEdx.s.Hi, X86_GREG_xDX);
8283 IEM_MC_REF_LOCAL(pu64EaxEdx, u64EaxEdx);
8284
8285 IEM_MC_FETCH_GREG_U32(u64EbxEcx.s.Lo, X86_GREG_xBX);
8286 IEM_MC_FETCH_GREG_U32(u64EbxEcx.s.Hi, X86_GREG_xCX);
8287 IEM_MC_REF_LOCAL(pu64EbxEcx, u64EbxEcx);
8288
8289 IEM_MC_FETCH_EFLAGS(EFlags);
8290 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8291 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg8b, pu64MemDst, pu64EaxEdx, pu64EbxEcx, pEFlags);
8292 else
8293 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg8b_locked, pu64MemDst, pu64EaxEdx, pu64EbxEcx, pEFlags);
8294
8295 IEM_MC_MEM_COMMIT_AND_UNMAP(pu64MemDst, IEM_ACCESS_DATA_RW);
8296 IEM_MC_COMMIT_EFLAGS(EFlags);
8297 IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_ZF)
8298 /** @todo Testcase: Check effect of cmpxchg8b on bits 63:32 in rax and rdx. */
8299 IEM_MC_STORE_GREG_U32(X86_GREG_xAX, u64EaxEdx.s.Lo);
8300 IEM_MC_STORE_GREG_U32(X86_GREG_xDX, u64EaxEdx.s.Hi);
8301 IEM_MC_ENDIF();
8302 IEM_MC_ADVANCE_RIP();
8303
8304 IEM_MC_END();
8305 return VINF_SUCCESS;
8306}
8307
8308
8309/** Opcode REX.W 0x0f 0xc7 !11/1. */
8310FNIEMOP_DEF_1(iemOp_Grp9_cmpxchg16b_Mdq, uint8_t, bRm)
8311{
8312 IEMOP_MNEMONIC(cmpxchg16b, "cmpxchg16b Mdq");
8313 if (IEM_GET_GUEST_CPU_FEATURES(pVCpu)->fMovCmpXchg16b)
8314 {
8315#if 0
8316 RT_NOREF(bRm);
8317 IEMOP_BITCH_ABOUT_STUB();
8318 return VERR_IEM_INSTR_NOT_IMPLEMENTED;
8319#else
8320 IEM_MC_BEGIN(4, 3);
8321 IEM_MC_ARG(PRTUINT128U, pu128MemDst, 0);
8322 IEM_MC_ARG(PRTUINT128U, pu128RaxRdx, 1);
8323 IEM_MC_ARG(PRTUINT128U, pu128RbxRcx, 2);
8324 IEM_MC_ARG_LOCAL_EFLAGS(pEFlags, EFlags, 3);
8325 IEM_MC_LOCAL(RTUINT128U, u128RaxRdx);
8326 IEM_MC_LOCAL(RTUINT128U, u128RbxRcx);
8327 IEM_MC_LOCAL(RTGCPTR, GCPtrEffDst);
8328
8329 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffDst, bRm, 0);
8330 IEMOP_HLP_DONE_DECODING();
8331 IEM_MC_RAISE_GP0_IF_EFF_ADDR_UNALIGNED(GCPtrEffDst, 16);
8332 IEM_MC_MEM_MAP(pu128MemDst, IEM_ACCESS_DATA_RW, pVCpu->iem.s.iEffSeg, GCPtrEffDst, 0 /*arg*/);
8333
8334 IEM_MC_FETCH_GREG_U64(u128RaxRdx.s.Lo, X86_GREG_xAX);
8335 IEM_MC_FETCH_GREG_U64(u128RaxRdx.s.Hi, X86_GREG_xDX);
8336 IEM_MC_REF_LOCAL(pu128RaxRdx, u128RaxRdx);
8337
8338 IEM_MC_FETCH_GREG_U64(u128RbxRcx.s.Lo, X86_GREG_xBX);
8339 IEM_MC_FETCH_GREG_U64(u128RbxRcx.s.Hi, X86_GREG_xCX);
8340 IEM_MC_REF_LOCAL(pu128RbxRcx, u128RbxRcx);
8341
8342 IEM_MC_FETCH_EFLAGS(EFlags);
8343# ifdef RT_ARCH_AMD64
8344 if (IEM_GET_HOST_CPU_FEATURES(pVCpu)->fMovCmpXchg16b)
8345 {
8346 if (!(pVCpu->iem.s.fPrefixes & IEM_OP_PRF_LOCK))
8347 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg16b, pu128MemDst, pu128RaxRdx, pu128RbxRcx, pEFlags);
8348 else
8349 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg16b_locked, pu128MemDst, pu128RaxRdx, pu128RbxRcx, pEFlags);
8350 }
8351 else
8352# endif
8353 {
8354 /* Note! The fallback for 32-bit systems and systems without CX16 is multiple
8355 accesses and not all all atomic, which works fine on in UNI CPU guest
8356 configuration (ignoring DMA). If guest SMP is active we have no choice
8357 but to use a rendezvous callback here. Sigh. */
8358 if (pVCpu->CTX_SUFF(pVM)->cCpus == 1)
8359 IEM_MC_CALL_VOID_AIMPL_4(iemAImpl_cmpxchg16b_fallback, pu128MemDst, pu128RaxRdx, pu128RbxRcx, pEFlags);
8360 else
8361 {
8362 IEM_MC_CALL_CIMPL_4(iemCImpl_cmpxchg16b_fallback_rendezvous, pu128MemDst, pu128RaxRdx, pu128RbxRcx, pEFlags);
8363 /* Does not get here, tail code is duplicated in iemCImpl_cmpxchg16b_fallback_rendezvous. */
8364 }
8365 }
8366
8367 IEM_MC_MEM_COMMIT_AND_UNMAP(pu128MemDst, IEM_ACCESS_DATA_RW);
8368 IEM_MC_COMMIT_EFLAGS(EFlags);
8369 IEM_MC_IF_EFL_BIT_NOT_SET(X86_EFL_ZF)
8370 IEM_MC_STORE_GREG_U64(X86_GREG_xAX, u128RaxRdx.s.Lo);
8371 IEM_MC_STORE_GREG_U64(X86_GREG_xDX, u128RaxRdx.s.Hi);
8372 IEM_MC_ENDIF();
8373 IEM_MC_ADVANCE_RIP();
8374
8375 IEM_MC_END();
8376 return VINF_SUCCESS;
8377#endif
8378 }
8379 Log(("cmpxchg16b -> #UD\n"));
8380 return IEMOP_RAISE_INVALID_OPCODE();
8381}
8382
8383FNIEMOP_DEF_1(iemOp_Grp9_cmpxchg8bOr16b, uint8_t, bRm)
8384{
8385 if (pVCpu->iem.s.fPrefixes & IEM_OP_PRF_SIZE_REX_W)
8386 return FNIEMOP_CALL_1(iemOp_Grp9_cmpxchg16b_Mdq, bRm);
8387 return FNIEMOP_CALL_1(iemOp_Grp9_cmpxchg8b_Mq, bRm);
8388}
8389
8390/** Opcode 0x0f 0xc7 11/6. */
8391FNIEMOP_UD_STUB_1(iemOp_Grp9_rdrand_Rv, uint8_t, bRm);
8392
8393/** Opcode 0x0f 0xc7 !11/6. */
8394FNIEMOP_UD_STUB_1(iemOp_Grp9_vmptrld_Mq, uint8_t, bRm);
8395
8396/** Opcode 0x66 0x0f 0xc7 !11/6. */
8397FNIEMOP_UD_STUB_1(iemOp_Grp9_vmclear_Mq, uint8_t, bRm);
8398
8399/** Opcode 0xf3 0x0f 0xc7 !11/6. */
8400FNIEMOP_UD_STUB_1(iemOp_Grp9_vmxon_Mq, uint8_t, bRm);
8401
8402/** Opcode [0xf3] 0x0f 0xc7 !11/7. */
8403FNIEMOP_UD_STUB_1(iemOp_Grp9_vmptrst_Mq, uint8_t, bRm);
8404
8405/** Opcode 0x0f 0xc7 11/7. */
8406FNIEMOP_UD_STUB_1(iemOp_Grp9_rdseed_Rv, uint8_t, bRm);
8407
8408
8409/**
8410 * Group 9 jump table for register variant.
8411 */
8412IEM_STATIC const PFNIEMOPRM g_apfnGroup9RegReg[] =
8413{ /* pfx: none, 066h, 0f3h, 0f2h */
8414 /* /0 */ IEMOP_X4(iemOp_InvalidWithRM),
8415 /* /1 */ IEMOP_X4(iemOp_InvalidWithRM),
8416 /* /2 */ IEMOP_X4(iemOp_InvalidWithRM),
8417 /* /3 */ IEMOP_X4(iemOp_InvalidWithRM),
8418 /* /4 */ IEMOP_X4(iemOp_InvalidWithRM),
8419 /* /5 */ IEMOP_X4(iemOp_InvalidWithRM),
8420 /* /6 */ iemOp_Grp9_rdrand_Rv, iemOp_Grp9_rdrand_Rv, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
8421 /* /7 */ iemOp_Grp9_rdseed_Rv, iemOp_Grp9_rdseed_Rv, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
8422};
8423AssertCompile(RT_ELEMENTS(g_apfnGroup9RegReg) == 8*4);
8424
8425
8426/**
8427 * Group 9 jump table for memory variant.
8428 */
8429IEM_STATIC const PFNIEMOPRM g_apfnGroup9MemReg[] =
8430{ /* pfx: none, 066h, 0f3h, 0f2h */
8431 /* /0 */ IEMOP_X4(iemOp_InvalidWithRM),
8432 /* /1 */ iemOp_Grp9_cmpxchg8bOr16b, iemOp_Grp9_cmpxchg8bOr16b, iemOp_Grp9_cmpxchg8bOr16b, iemOp_Grp9_cmpxchg8bOr16b, /* see bs3-cpu-decoding-1 */
8433 /* /2 */ IEMOP_X4(iemOp_InvalidWithRM),
8434 /* /3 */ IEMOP_X4(iemOp_InvalidWithRM),
8435 /* /4 */ IEMOP_X4(iemOp_InvalidWithRM),
8436 /* /5 */ IEMOP_X4(iemOp_InvalidWithRM),
8437 /* /6 */ iemOp_Grp9_vmptrld_Mq, iemOp_Grp9_vmclear_Mq, iemOp_Grp9_vmxon_Mq, iemOp_InvalidWithRM,
8438 /* /7 */ iemOp_Grp9_vmptrst_Mq, iemOp_InvalidWithRM, iemOp_InvalidWithRM, iemOp_InvalidWithRM,
8439};
8440AssertCompile(RT_ELEMENTS(g_apfnGroup9MemReg) == 8*4);
8441
8442
8443/** Opcode 0x0f 0xc7. */
8444FNIEMOP_DEF(iemOp_Grp9)
8445{
8446 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8447 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
8448 /* register, register */
8449 return FNIEMOP_CALL_1(g_apfnGroup9RegReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
8450 + pVCpu->iem.s.idxPrefix], bRm);
8451 /* memory, register */
8452 return FNIEMOP_CALL_1(g_apfnGroup9MemReg[ ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) * 4
8453 + pVCpu->iem.s.idxPrefix], bRm);
8454}
8455
8456
8457/**
8458 * Common 'bswap register' helper.
8459 */
8460FNIEMOP_DEF_1(iemOpCommonBswapGReg, uint8_t, iReg)
8461{
8462 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8463 switch (pVCpu->iem.s.enmEffOpSize)
8464 {
8465 case IEMMODE_16BIT:
8466 IEM_MC_BEGIN(1, 0);
8467 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
8468 IEM_MC_REF_GREG_U32(pu32Dst, iReg); /* Don't clear the high dword! */
8469 IEM_MC_CALL_VOID_AIMPL_1(iemAImpl_bswap_u16, pu32Dst);
8470 IEM_MC_ADVANCE_RIP();
8471 IEM_MC_END();
8472 return VINF_SUCCESS;
8473
8474 case IEMMODE_32BIT:
8475 IEM_MC_BEGIN(1, 0);
8476 IEM_MC_ARG(uint32_t *, pu32Dst, 0);
8477 IEM_MC_REF_GREG_U32(pu32Dst, iReg);
8478 IEM_MC_CLEAR_HIGH_GREG_U64_BY_REF(pu32Dst);
8479 IEM_MC_CALL_VOID_AIMPL_1(iemAImpl_bswap_u32, pu32Dst);
8480 IEM_MC_ADVANCE_RIP();
8481 IEM_MC_END();
8482 return VINF_SUCCESS;
8483
8484 case IEMMODE_64BIT:
8485 IEM_MC_BEGIN(1, 0);
8486 IEM_MC_ARG(uint64_t *, pu64Dst, 0);
8487 IEM_MC_REF_GREG_U64(pu64Dst, iReg);
8488 IEM_MC_CALL_VOID_AIMPL_1(iemAImpl_bswap_u64, pu64Dst);
8489 IEM_MC_ADVANCE_RIP();
8490 IEM_MC_END();
8491 return VINF_SUCCESS;
8492
8493 IEM_NOT_REACHED_DEFAULT_CASE_RET();
8494 }
8495}
8496
8497
8498/** Opcode 0x0f 0xc8. */
8499FNIEMOP_DEF(iemOp_bswap_rAX_r8)
8500{
8501 IEMOP_MNEMONIC(bswap_rAX_r8, "bswap rAX/r8");
8502 /* Note! Intel manuals states that R8-R15 can be accessed by using a REX.X
8503 prefix. REX.B is the correct prefix it appears. For a parallel
8504 case, see iemOp_mov_AL_Ib and iemOp_mov_eAX_Iv. */
8505 IEMOP_HLP_MIN_486();
8506 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xAX | pVCpu->iem.s.uRexB);
8507}
8508
8509
8510/** Opcode 0x0f 0xc9. */
8511FNIEMOP_DEF(iemOp_bswap_rCX_r9)
8512{
8513 IEMOP_MNEMONIC(bswap_rCX_r9, "bswap rCX/r9");
8514 IEMOP_HLP_MIN_486();
8515 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xCX | pVCpu->iem.s.uRexB);
8516}
8517
8518
8519/** Opcode 0x0f 0xca. */
8520FNIEMOP_DEF(iemOp_bswap_rDX_r10)
8521{
8522 IEMOP_MNEMONIC(bswap_rDX_r9, "bswap rDX/r9");
8523 IEMOP_HLP_MIN_486();
8524 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xDX | pVCpu->iem.s.uRexB);
8525}
8526
8527
8528/** Opcode 0x0f 0xcb. */
8529FNIEMOP_DEF(iemOp_bswap_rBX_r11)
8530{
8531 IEMOP_MNEMONIC(bswap_rBX_r9, "bswap rBX/r9");
8532 IEMOP_HLP_MIN_486();
8533 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xBX | pVCpu->iem.s.uRexB);
8534}
8535
8536
8537/** Opcode 0x0f 0xcc. */
8538FNIEMOP_DEF(iemOp_bswap_rSP_r12)
8539{
8540 IEMOP_MNEMONIC(bswap_rSP_r12, "bswap rSP/r12");
8541 IEMOP_HLP_MIN_486();
8542 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xSP | pVCpu->iem.s.uRexB);
8543}
8544
8545
8546/** Opcode 0x0f 0xcd. */
8547FNIEMOP_DEF(iemOp_bswap_rBP_r13)
8548{
8549 IEMOP_MNEMONIC(bswap_rBP_r13, "bswap rBP/r13");
8550 IEMOP_HLP_MIN_486();
8551 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xBP | pVCpu->iem.s.uRexB);
8552}
8553
8554
8555/** Opcode 0x0f 0xce. */
8556FNIEMOP_DEF(iemOp_bswap_rSI_r14)
8557{
8558 IEMOP_MNEMONIC(bswap_rSI_r14, "bswap rSI/r14");
8559 IEMOP_HLP_MIN_486();
8560 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xSI | pVCpu->iem.s.uRexB);
8561}
8562
8563
8564/** Opcode 0x0f 0xcf. */
8565FNIEMOP_DEF(iemOp_bswap_rDI_r15)
8566{
8567 IEMOP_MNEMONIC(bswap_rDI_r15, "bswap rDI/r15");
8568 IEMOP_HLP_MIN_486();
8569 return FNIEMOP_CALL_1(iemOpCommonBswapGReg, X86_GREG_xDI | pVCpu->iem.s.uRexB);
8570}
8571
8572
8573/* Opcode 0x0f 0xd0 - invalid */
8574/** Opcode 0x66 0x0f 0xd0 - addsubpd Vpd, Wpd */
8575FNIEMOP_STUB(iemOp_addsubpd_Vpd_Wpd);
8576/* Opcode 0xf3 0x0f 0xd0 - invalid */
8577/** Opcode 0xf2 0x0f 0xd0 - addsubps Vps, Wps */
8578FNIEMOP_STUB(iemOp_addsubps_Vps_Wps);
8579
8580/** Opcode 0x0f 0xd1 - psrlw Pq, Qq */
8581FNIEMOP_STUB(iemOp_psrlw_Pq_Qq);
8582/** Opcode 0x66 0x0f 0xd1 - psrlw Vx, W */
8583FNIEMOP_STUB(iemOp_psrlw_Vx_W);
8584/* Opcode 0xf3 0x0f 0xd1 - invalid */
8585/* Opcode 0xf2 0x0f 0xd1 - invalid */
8586
8587/** Opcode 0x0f 0xd2 - psrld Pq, Qq */
8588FNIEMOP_STUB(iemOp_psrld_Pq_Qq);
8589/** Opcode 0x66 0x0f 0xd2 - psrld Vx, Wx */
8590FNIEMOP_STUB(iemOp_psrld_Vx_Wx);
8591/* Opcode 0xf3 0x0f 0xd2 - invalid */
8592/* Opcode 0xf2 0x0f 0xd2 - invalid */
8593
8594/** Opcode 0x0f 0xd3 - psrlq Pq, Qq */
8595FNIEMOP_STUB(iemOp_psrlq_Pq_Qq);
8596/** Opcode 0x66 0x0f 0xd3 - psrlq Vx, Wx */
8597FNIEMOP_STUB(iemOp_psrlq_Vx_Wx);
8598/* Opcode 0xf3 0x0f 0xd3 - invalid */
8599/* Opcode 0xf2 0x0f 0xd3 - invalid */
8600
8601/** Opcode 0x0f 0xd4 - paddq Pq, Qq */
8602FNIEMOP_STUB(iemOp_paddq_Pq_Qq);
8603/** Opcode 0x66 0x0f 0xd4 - paddq Vx, W */
8604FNIEMOP_STUB(iemOp_paddq_Vx_W);
8605/* Opcode 0xf3 0x0f 0xd4 - invalid */
8606/* Opcode 0xf2 0x0f 0xd4 - invalid */
8607
8608/** Opcode 0x0f 0xd5 - pmullw Pq, Qq */
8609FNIEMOP_STUB(iemOp_pmullw_Pq_Qq);
8610/** Opcode 0x66 0x0f 0xd5 - pmullw Vx, Wx */
8611FNIEMOP_STUB(iemOp_pmullw_Vx_Wx);
8612/* Opcode 0xf3 0x0f 0xd5 - invalid */
8613/* Opcode 0xf2 0x0f 0xd5 - invalid */
8614
8615/* Opcode 0x0f 0xd6 - invalid */
8616
8617/**
8618 * @opcode 0xd6
8619 * @oppfx 0x66
8620 * @opcpuid sse2
8621 * @opgroup og_sse2_pcksclr_datamove
8622 * @opxcpttype none
8623 * @optest op1=-1 op2=2 -> op1=2
8624 * @optest op1=0 op2=-42 -> op1=-42
8625 */
8626FNIEMOP_DEF(iemOp_movq_Wq_Vq)
8627{
8628 IEMOP_MNEMONIC2(MR, MOVQ, movq, WqZxReg_WO, Vq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
8629 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8630 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
8631 {
8632 /*
8633 * Register, register.
8634 */
8635 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8636 IEM_MC_BEGIN(0, 2);
8637 IEM_MC_LOCAL(uint64_t, uSrc);
8638
8639 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
8640 IEM_MC_ACTUALIZE_SSE_STATE_FOR_CHANGE();
8641
8642 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8643 IEM_MC_STORE_XREG_U64_ZX_U128((bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB, uSrc);
8644
8645 IEM_MC_ADVANCE_RIP();
8646 IEM_MC_END();
8647 }
8648 else
8649 {
8650 /*
8651 * Memory, register.
8652 */
8653 IEM_MC_BEGIN(0, 2);
8654 IEM_MC_LOCAL(uint64_t, uSrc);
8655 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
8656
8657 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
8658 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8659 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
8660 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
8661
8662 IEM_MC_FETCH_XREG_U64(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8663 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
8664
8665 IEM_MC_ADVANCE_RIP();
8666 IEM_MC_END();
8667 }
8668 return VINF_SUCCESS;
8669}
8670
8671
8672/**
8673 * @opcode 0xd6
8674 * @opcodesub 11 mr/reg
8675 * @oppfx f3
8676 * @opcpuid sse2
8677 * @opgroup og_sse2_simdint_datamove
8678 * @optest op1=1 op2=2 -> op1=2 ftw=0xff
8679 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
8680 */
8681FNIEMOP_DEF(iemOp_movq2dq_Vdq_Nq)
8682{
8683 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8684 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
8685 {
8686 /*
8687 * Register, register.
8688 */
8689 IEMOP_MNEMONIC2(RM_REG, MOVQ2DQ, movq2dq, VqZx_WO, Nq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
8690 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8691 IEM_MC_BEGIN(0, 1);
8692 IEM_MC_LOCAL(uint64_t, uSrc);
8693
8694 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
8695 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
8696
8697 IEM_MC_FETCH_MREG_U64(uSrc, bRm & X86_MODRM_RM_MASK);
8698 IEM_MC_STORE_XREG_U64_ZX_U128(((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg, uSrc);
8699 IEM_MC_FPU_TO_MMX_MODE();
8700
8701 IEM_MC_ADVANCE_RIP();
8702 IEM_MC_END();
8703 return VINF_SUCCESS;
8704 }
8705
8706 /**
8707 * @opdone
8708 * @opmnemonic udf30fd6mem
8709 * @opcode 0xd6
8710 * @opcodesub !11 mr/reg
8711 * @oppfx f3
8712 * @opunused intel-modrm
8713 * @opcpuid sse
8714 * @optest ->
8715 */
8716 return FNIEMOP_CALL_1(iemOp_InvalidWithRMNeedDecode, bRm);
8717}
8718
8719
8720/**
8721 * @opcode 0xd6
8722 * @opcodesub 11 mr/reg
8723 * @oppfx f2
8724 * @opcpuid sse2
8725 * @opgroup og_sse2_simdint_datamove
8726 * @optest op1=1 op2=2 -> op1=2 ftw=0xff
8727 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
8728 * @optest op1=0 op2=0x1123456789abcdef -> op1=0x1123456789abcdef ftw=0xff
8729 * @optest op1=0 op2=0xfedcba9876543210 -> op1=0xfedcba9876543210 ftw=0xff
8730 * @optest op1=-42 op2=0xfedcba9876543210
8731 * -> op1=0xfedcba9876543210 ftw=0xff
8732 */
8733FNIEMOP_DEF(iemOp_movdq2q_Pq_Uq)
8734{
8735 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8736 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT))
8737 {
8738 /*
8739 * Register, register.
8740 */
8741 IEMOP_MNEMONIC2(RM_REG, MOVDQ2Q, movdq2q, Pq_WO, Uq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
8742 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8743 IEM_MC_BEGIN(0, 1);
8744 IEM_MC_LOCAL(uint64_t, uSrc);
8745
8746 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
8747 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
8748
8749 IEM_MC_FETCH_XREG_U64(uSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
8750 IEM_MC_STORE_MREG_U64((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK, uSrc);
8751 IEM_MC_FPU_TO_MMX_MODE();
8752
8753 IEM_MC_ADVANCE_RIP();
8754 IEM_MC_END();
8755 return VINF_SUCCESS;
8756 }
8757
8758 /**
8759 * @opdone
8760 * @opmnemonic udf20fd6mem
8761 * @opcode 0xd6
8762 * @opcodesub !11 mr/reg
8763 * @oppfx f2
8764 * @opunused intel-modrm
8765 * @opcpuid sse
8766 * @optest ->
8767 */
8768 return FNIEMOP_CALL_1(iemOp_InvalidWithRMNeedDecode, bRm);
8769}
8770
8771/** Opcode 0x0f 0xd7 - pmovmskb Gd, Nq */
8772FNIEMOP_DEF(iemOp_pmovmskb_Gd_Nq)
8773{
8774 /* Note! Taking the lazy approch here wrt the high 32-bits of the GREG. */
8775 /** @todo testcase: Check that the instruction implicitly clears the high
8776 * bits in 64-bit mode. The REX.W is first necessary when VLMAX > 256
8777 * and opcode modifications are made to work with the whole width (not
8778 * just 128). */
8779 IEMOP_MNEMONIC(pmovmskb_Gd_Udq, "pmovmskb Gd,Nq");
8780 /* Docs says register only. */
8781 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8782 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT)) /** @todo test that this is registers only. */
8783 {
8784 IEMOP_HLP_DECODED_NL_2(OP_PMOVMSKB, IEMOPFORM_RM_REG, OP_PARM_Gd, OP_PARM_Vdq, DISOPTYPE_MMX | DISOPTYPE_HARMLESS);
8785 IEM_MC_BEGIN(2, 0);
8786 IEM_MC_ARG(uint64_t *, pDst, 0);
8787 IEM_MC_ARG(uint64_t const *, pSrc, 1);
8788 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT_CHECK_SSE_OR_MMXEXT();
8789 IEM_MC_PREPARE_FPU_USAGE();
8790 IEM_MC_REF_GREG_U64(pDst, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
8791 IEM_MC_REF_MREG_U64_CONST(pSrc, bRm & X86_MODRM_RM_MASK);
8792 IEM_MC_CALL_MMX_AIMPL_2(iemAImpl_pmovmskb_u64, pDst, pSrc);
8793 IEM_MC_ADVANCE_RIP();
8794 IEM_MC_END();
8795 return VINF_SUCCESS;
8796 }
8797 return IEMOP_RAISE_INVALID_OPCODE();
8798}
8799
8800/** Opcode 0x66 0x0f 0xd7 - */
8801FNIEMOP_DEF(iemOp_pmovmskb_Gd_Ux)
8802{
8803 /* Note! Taking the lazy approch here wrt the high 32-bits of the GREG. */
8804 /** @todo testcase: Check that the instruction implicitly clears the high
8805 * bits in 64-bit mode. The REX.W is first necessary when VLMAX > 256
8806 * and opcode modifications are made to work with the whole width (not
8807 * just 128). */
8808 IEMOP_MNEMONIC(pmovmskb_Gd_Nq, "vpmovmskb Gd, Ux");
8809 /* Docs says register only. */
8810 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8811 if ((bRm & X86_MODRM_MOD_MASK) == (3 << X86_MODRM_MOD_SHIFT)) /** @todo test that this is registers only. */
8812 {
8813 IEMOP_HLP_DECODED_NL_2(OP_PMOVMSKB, IEMOPFORM_RM_REG, OP_PARM_Gd, OP_PARM_Vdq, DISOPTYPE_SSE | DISOPTYPE_HARMLESS);
8814 IEM_MC_BEGIN(2, 0);
8815 IEM_MC_ARG(uint64_t *, pDst, 0);
8816 IEM_MC_ARG(PCRTUINT128U, pSrc, 1);
8817 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
8818 IEM_MC_PREPARE_SSE_USAGE();
8819 IEM_MC_REF_GREG_U64(pDst, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
8820 IEM_MC_REF_XREG_U128_CONST(pSrc, (bRm & X86_MODRM_RM_MASK) | pVCpu->iem.s.uRexB);
8821 IEM_MC_CALL_SSE_AIMPL_2(iemAImpl_pmovmskb_u128, pDst, pSrc);
8822 IEM_MC_ADVANCE_RIP();
8823 IEM_MC_END();
8824 return VINF_SUCCESS;
8825 }
8826 return IEMOP_RAISE_INVALID_OPCODE();
8827}
8828
8829/* Opcode 0xf3 0x0f 0xd7 - invalid */
8830/* Opcode 0xf2 0x0f 0xd7 - invalid */
8831
8832
8833/** Opcode 0x0f 0xd8 - psubusb Pq, Qq */
8834FNIEMOP_STUB(iemOp_psubusb_Pq_Qq);
8835/** Opcode 0x66 0x0f 0xd8 - psubusb Vx, W */
8836FNIEMOP_STUB(iemOp_psubusb_Vx_W);
8837/* Opcode 0xf3 0x0f 0xd8 - invalid */
8838/* Opcode 0xf2 0x0f 0xd8 - invalid */
8839
8840/** Opcode 0x0f 0xd9 - psubusw Pq, Qq */
8841FNIEMOP_STUB(iemOp_psubusw_Pq_Qq);
8842/** Opcode 0x66 0x0f 0xd9 - psubusw Vx, Wx */
8843FNIEMOP_STUB(iemOp_psubusw_Vx_Wx);
8844/* Opcode 0xf3 0x0f 0xd9 - invalid */
8845/* Opcode 0xf2 0x0f 0xd9 - invalid */
8846
8847/** Opcode 0x0f 0xda - pminub Pq, Qq */
8848FNIEMOP_STUB(iemOp_pminub_Pq_Qq);
8849/** Opcode 0x66 0x0f 0xda - pminub Vx, Wx */
8850FNIEMOP_STUB(iemOp_pminub_Vx_Wx);
8851/* Opcode 0xf3 0x0f 0xda - invalid */
8852/* Opcode 0xf2 0x0f 0xda - invalid */
8853
8854/** Opcode 0x0f 0xdb - pand Pq, Qq */
8855FNIEMOP_STUB(iemOp_pand_Pq_Qq);
8856/** Opcode 0x66 0x0f 0xdb - pand Vx, W */
8857FNIEMOP_STUB(iemOp_pand_Vx_W);
8858/* Opcode 0xf3 0x0f 0xdb - invalid */
8859/* Opcode 0xf2 0x0f 0xdb - invalid */
8860
8861/** Opcode 0x0f 0xdc - paddusb Pq, Qq */
8862FNIEMOP_STUB(iemOp_paddusb_Pq_Qq);
8863/** Opcode 0x66 0x0f 0xdc - paddusb Vx, Wx */
8864FNIEMOP_STUB(iemOp_paddusb_Vx_Wx);
8865/* Opcode 0xf3 0x0f 0xdc - invalid */
8866/* Opcode 0xf2 0x0f 0xdc - invalid */
8867
8868/** Opcode 0x0f 0xdd - paddusw Pq, Qq */
8869FNIEMOP_STUB(iemOp_paddusw_Pq_Qq);
8870/** Opcode 0x66 0x0f 0xdd - paddusw Vx, Wx */
8871FNIEMOP_STUB(iemOp_paddusw_Vx_Wx);
8872/* Opcode 0xf3 0x0f 0xdd - invalid */
8873/* Opcode 0xf2 0x0f 0xdd - invalid */
8874
8875/** Opcode 0x0f 0xde - pmaxub Pq, Qq */
8876FNIEMOP_STUB(iemOp_pmaxub_Pq_Qq);
8877/** Opcode 0x66 0x0f 0xde - pmaxub Vx, W */
8878FNIEMOP_STUB(iemOp_pmaxub_Vx_W);
8879/* Opcode 0xf3 0x0f 0xde - invalid */
8880/* Opcode 0xf2 0x0f 0xde - invalid */
8881
8882/** Opcode 0x0f 0xdf - pandn Pq, Qq */
8883FNIEMOP_STUB(iemOp_pandn_Pq_Qq);
8884/** Opcode 0x66 0x0f 0xdf - pandn Vx, Wx */
8885FNIEMOP_STUB(iemOp_pandn_Vx_Wx);
8886/* Opcode 0xf3 0x0f 0xdf - invalid */
8887/* Opcode 0xf2 0x0f 0xdf - invalid */
8888
8889/** Opcode 0x0f 0xe0 - pavgb Pq, Qq */
8890FNIEMOP_STUB(iemOp_pavgb_Pq_Qq);
8891/** Opcode 0x66 0x0f 0xe0 - pavgb Vx, Wx */
8892FNIEMOP_STUB(iemOp_pavgb_Vx_Wx);
8893/* Opcode 0xf3 0x0f 0xe0 - invalid */
8894/* Opcode 0xf2 0x0f 0xe0 - invalid */
8895
8896/** Opcode 0x0f 0xe1 - psraw Pq, Qq */
8897FNIEMOP_STUB(iemOp_psraw_Pq_Qq);
8898/** Opcode 0x66 0x0f 0xe1 - psraw Vx, W */
8899FNIEMOP_STUB(iemOp_psraw_Vx_W);
8900/* Opcode 0xf3 0x0f 0xe1 - invalid */
8901/* Opcode 0xf2 0x0f 0xe1 - invalid */
8902
8903/** Opcode 0x0f 0xe2 - psrad Pq, Qq */
8904FNIEMOP_STUB(iemOp_psrad_Pq_Qq);
8905/** Opcode 0x66 0x0f 0xe2 - psrad Vx, Wx */
8906FNIEMOP_STUB(iemOp_psrad_Vx_Wx);
8907/* Opcode 0xf3 0x0f 0xe2 - invalid */
8908/* Opcode 0xf2 0x0f 0xe2 - invalid */
8909
8910/** Opcode 0x0f 0xe3 - pavgw Pq, Qq */
8911FNIEMOP_STUB(iemOp_pavgw_Pq_Qq);
8912/** Opcode 0x66 0x0f 0xe3 - pavgw Vx, Wx */
8913FNIEMOP_STUB(iemOp_pavgw_Vx_Wx);
8914/* Opcode 0xf3 0x0f 0xe3 - invalid */
8915/* Opcode 0xf2 0x0f 0xe3 - invalid */
8916
8917/** Opcode 0x0f 0xe4 - pmulhuw Pq, Qq */
8918FNIEMOP_STUB(iemOp_pmulhuw_Pq_Qq);
8919/** Opcode 0x66 0x0f 0xe4 - pmulhuw Vx, W */
8920FNIEMOP_STUB(iemOp_pmulhuw_Vx_W);
8921/* Opcode 0xf3 0x0f 0xe4 - invalid */
8922/* Opcode 0xf2 0x0f 0xe4 - invalid */
8923
8924/** Opcode 0x0f 0xe5 - pmulhw Pq, Qq */
8925FNIEMOP_STUB(iemOp_pmulhw_Pq_Qq);
8926/** Opcode 0x66 0x0f 0xe5 - pmulhw Vx, Wx */
8927FNIEMOP_STUB(iemOp_pmulhw_Vx_Wx);
8928/* Opcode 0xf3 0x0f 0xe5 - invalid */
8929/* Opcode 0xf2 0x0f 0xe5 - invalid */
8930
8931/* Opcode 0x0f 0xe6 - invalid */
8932/** Opcode 0x66 0x0f 0xe6 - cvttpd2dq Vx, Wpd */
8933FNIEMOP_STUB(iemOp_cvttpd2dq_Vx_Wpd);
8934/** Opcode 0xf3 0x0f 0xe6 - cvtdq2pd Vx, Wpd */
8935FNIEMOP_STUB(iemOp_cvtdq2pd_Vx_Wpd);
8936/** Opcode 0xf2 0x0f 0xe6 - cvtpd2dq Vx, Wpd */
8937FNIEMOP_STUB(iemOp_cvtpd2dq_Vx_Wpd);
8938
8939
8940/**
8941 * @opcode 0xe7
8942 * @opcodesub !11 mr/reg
8943 * @oppfx none
8944 * @opcpuid sse
8945 * @opgroup og_sse1_cachect
8946 * @opxcpttype none
8947 * @optest op1=-1 op2=2 -> op1=2 ftw=0xff
8948 * @optest op1=0 op2=-42 -> op1=-42 ftw=0xff
8949 */
8950FNIEMOP_DEF(iemOp_movntq_Mq_Pq)
8951{
8952 IEMOP_MNEMONIC2(MR_MEM, MOVNTQ, movntq, Mq_WO, Pq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
8953 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
8954 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
8955 {
8956 /* Register, memory. */
8957 IEM_MC_BEGIN(0, 2);
8958 IEM_MC_LOCAL(uint64_t, uSrc);
8959 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
8960
8961 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
8962 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
8963 IEM_MC_MAYBE_RAISE_MMX_RELATED_XCPT();
8964 IEM_MC_ACTUALIZE_FPU_STATE_FOR_CHANGE();
8965
8966 IEM_MC_FETCH_MREG_U64(uSrc, (bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK);
8967 IEM_MC_STORE_MEM_U64(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
8968 IEM_MC_FPU_TO_MMX_MODE();
8969
8970 IEM_MC_ADVANCE_RIP();
8971 IEM_MC_END();
8972 return VINF_SUCCESS;
8973 }
8974 /**
8975 * @opdone
8976 * @opmnemonic ud0fe7reg
8977 * @opcode 0xe7
8978 * @opcodesub 11 mr/reg
8979 * @oppfx none
8980 * @opunused immediate
8981 * @opcpuid sse
8982 * @optest ->
8983 */
8984 return IEMOP_RAISE_INVALID_OPCODE();
8985}
8986
8987/**
8988 * @opcode 0xe7
8989 * @opcodesub !11 mr/reg
8990 * @oppfx 0x66
8991 * @opcpuid sse2
8992 * @opgroup og_sse2_cachect
8993 * @opxcpttype 1
8994 * @optest op1=-1 op2=2 -> op1=2
8995 * @optest op1=0 op2=-42 -> op1=-42
8996 */
8997FNIEMOP_DEF(iemOp_movntdq_Mdq_Vdq)
8998{
8999 IEMOP_MNEMONIC2(MR_MEM, MOVNTDQ, movntdq, Mdq_WO, Vdq, DISOPTYPE_HARMLESS, IEMOPHINT_IGNORES_OP_SIZES);
9000 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm);
9001 if ((bRm & X86_MODRM_MOD_MASK) != (3 << X86_MODRM_MOD_SHIFT))
9002 {
9003 /* Register, memory. */
9004 IEM_MC_BEGIN(0, 2);
9005 IEM_MC_LOCAL(RTUINT128U, uSrc);
9006 IEM_MC_LOCAL(RTGCPTR, GCPtrEffSrc);
9007
9008 IEM_MC_CALC_RM_EFF_ADDR(GCPtrEffSrc, bRm, 0);
9009 IEMOP_HLP_DONE_DECODING_NO_LOCK_PREFIX();
9010 IEM_MC_MAYBE_RAISE_SSE2_RELATED_XCPT();
9011 IEM_MC_ACTUALIZE_SSE_STATE_FOR_READ();
9012
9013 IEM_MC_FETCH_XREG_U128(uSrc, ((bRm >> X86_MODRM_REG_SHIFT) & X86_MODRM_REG_SMASK) | pVCpu->iem.s.uRexReg);
9014 IEM_MC_STORE_MEM_U128_ALIGN_SSE(pVCpu->iem.s.iEffSeg, GCPtrEffSrc, uSrc);
9015
9016 IEM_MC_ADVANCE_RIP();
9017 IEM_MC_END();
9018 return VINF_SUCCESS;
9019 }
9020
9021 /**
9022 * @opdone
9023 * @opmnemonic ud660fe7reg
9024 * @opcode 0xe7
9025 * @opcodesub 11 mr/reg
9026 * @oppfx 0x66
9027 * @opunused immediate
9028 * @opcpuid sse
9029 * @optest ->
9030 */
9031 return IEMOP_RAISE_INVALID_OPCODE();
9032}
9033
9034/* Opcode 0xf3 0x0f 0xe7 - invalid */
9035/* Opcode 0xf2 0x0f 0xe7 - invalid */
9036
9037
9038/** Opcode 0x0f 0xe8 - psubsb Pq, Qq */
9039FNIEMOP_STUB(iemOp_psubsb_Pq_Qq);
9040/** Opcode 0x66 0x0f 0xe8 - psubsb Vx, W */
9041FNIEMOP_STUB(iemOp_psubsb_Vx_W);
9042/* Opcode 0xf3 0x0f 0xe8 - invalid */
9043/* Opcode 0xf2 0x0f 0xe8 - invalid */
9044
9045/** Opcode 0x0f 0xe9 - psubsw Pq, Qq */
9046FNIEMOP_STUB(iemOp_psubsw_Pq_Qq);
9047/** Opcode 0x66 0x0f 0xe9 - psubsw Vx, Wx */
9048FNIEMOP_STUB(iemOp_psubsw_Vx_Wx);
9049/* Opcode 0xf3 0x0f 0xe9 - invalid */
9050/* Opcode 0xf2 0x0f 0xe9 - invalid */
9051
9052/** Opcode 0x0f 0xea - pminsw Pq, Qq */
9053FNIEMOP_STUB(iemOp_pminsw_Pq_Qq);
9054/** Opcode 0x66 0x0f 0xea - pminsw Vx, Wx */
9055FNIEMOP_STUB(iemOp_pminsw_Vx_Wx);
9056/* Opcode 0xf3 0x0f 0xea - invalid */
9057/* Opcode 0xf2 0x0f 0xea - invalid */
9058
9059/** Opcode 0x0f 0xeb - por Pq, Qq */
9060FNIEMOP_STUB(iemOp_por_Pq_Qq);
9061/** Opcode 0x66 0x0f 0xeb - por Vx, W */
9062FNIEMOP_STUB(iemOp_por_Vx_W);
9063/* Opcode 0xf3 0x0f 0xeb - invalid */
9064/* Opcode 0xf2 0x0f 0xeb - invalid */
9065
9066/** Opcode 0x0f 0xec - paddsb Pq, Qq */
9067FNIEMOP_STUB(iemOp_paddsb_Pq_Qq);
9068/** Opcode 0x66 0x0f 0xec - paddsb Vx, Wx */
9069FNIEMOP_STUB(iemOp_paddsb_Vx_Wx);
9070/* Opcode 0xf3 0x0f 0xec - invalid */
9071/* Opcode 0xf2 0x0f 0xec - invalid */
9072
9073/** Opcode 0x0f 0xed - paddsw Pq, Qq */
9074FNIEMOP_STUB(iemOp_paddsw_Pq_Qq);
9075/** Opcode 0x66 0x0f 0xed - paddsw Vx, Wx */
9076FNIEMOP_STUB(iemOp_paddsw_Vx_Wx);
9077/* Opcode 0xf3 0x0f 0xed - invalid */
9078/* Opcode 0xf2 0x0f 0xed - invalid */
9079
9080/** Opcode 0x0f 0xee - pmaxsw Pq, Qq */
9081FNIEMOP_STUB(iemOp_pmaxsw_Pq_Qq);
9082/** Opcode 0x66 0x0f 0xee - pmaxsw Vx, W */
9083FNIEMOP_STUB(iemOp_pmaxsw_Vx_W);
9084/* Opcode 0xf3 0x0f 0xee - invalid */
9085/* Opcode 0xf2 0x0f 0xee - invalid */
9086
9087
9088/** Opcode 0x0f 0xef - pxor Pq, Qq */
9089FNIEMOP_DEF(iemOp_pxor_Pq_Qq)
9090{
9091 IEMOP_MNEMONIC(pxor, "pxor");
9092 return FNIEMOP_CALL_1(iemOpCommonMmx_FullFull_To_Full, &g_iemAImpl_pxor);
9093}
9094
9095/** Opcode 0x66 0x0f 0xef - pxor Vx, Wx */
9096FNIEMOP_DEF(iemOp_pxor_Vx_Wx)
9097{
9098 IEMOP_MNEMONIC(pxor_Vx_Wx, "pxor");
9099 return FNIEMOP_CALL_1(iemOpCommonSse2_FullFull_To_Full, &g_iemAImpl_pxor);
9100}
9101
9102/* Opcode 0xf3 0x0f 0xef - invalid */
9103/* Opcode 0xf2 0x0f 0xef - invalid */
9104
9105/* Opcode 0x0f 0xf0 - invalid */
9106/* Opcode 0x66 0x0f 0xf0 - invalid */
9107/** Opcode 0xf2 0x0f 0xf0 - lddqu Vx, Mx */
9108FNIEMOP_STUB(iemOp_lddqu_Vx_Mx);
9109
9110/** Opcode 0x0f 0xf1 - psllw Pq, Qq */
9111FNIEMOP_STUB(iemOp_psllw_Pq_Qq);
9112/** Opcode 0x66 0x0f 0xf1 - psllw Vx, W */
9113FNIEMOP_STUB(iemOp_psllw_Vx_W);
9114/* Opcode 0xf2 0x0f 0xf1 - invalid */
9115
9116/** Opcode 0x0f 0xf2 - pslld Pq, Qq */
9117FNIEMOP_STUB(iemOp_pslld_Pq_Qq);
9118/** Opcode 0x66 0x0f 0xf2 - pslld Vx, Wx */
9119FNIEMOP_STUB(iemOp_pslld_Vx_Wx);
9120/* Opcode 0xf2 0x0f 0xf2 - invalid */
9121
9122/** Opcode 0x0f 0xf3 - psllq Pq, Qq */
9123FNIEMOP_STUB(iemOp_psllq_Pq_Qq);
9124/** Opcode 0x66 0x0f 0xf3 - psllq Vx, Wx */
9125FNIEMOP_STUB(iemOp_psllq_Vx_Wx);
9126/* Opcode 0xf2 0x0f 0xf3 - invalid */
9127
9128/** Opcode 0x0f 0xf4 - pmuludq Pq, Qq */
9129FNIEMOP_STUB(iemOp_pmuludq_Pq_Qq);
9130/** Opcode 0x66 0x0f 0xf4 - pmuludq Vx, W */
9131FNIEMOP_STUB(iemOp_pmuludq_Vx_W);
9132/* Opcode 0xf2 0x0f 0xf4 - invalid */
9133
9134/** Opcode 0x0f 0xf5 - pmaddwd Pq, Qq */
9135FNIEMOP_STUB(iemOp_pmaddwd_Pq_Qq);
9136/** Opcode 0x66 0x0f 0xf5 - pmaddwd Vx, Wx */
9137FNIEMOP_STUB(iemOp_pmaddwd_Vx_Wx);
9138/* Opcode 0xf2 0x0f 0xf5 - invalid */
9139
9140/** Opcode 0x0f 0xf6 - psadbw Pq, Qq */
9141FNIEMOP_STUB(iemOp_psadbw_Pq_Qq);
9142/** Opcode 0x66 0x0f 0xf6 - psadbw Vx, Wx */
9143FNIEMOP_STUB(iemOp_psadbw_Vx_Wx);
9144/* Opcode 0xf2 0x0f 0xf6 - invalid */
9145
9146/** Opcode 0x0f 0xf7 - maskmovq Pq, Nq */
9147FNIEMOP_STUB(iemOp_maskmovq_Pq_Nq);
9148/** Opcode 0x66 0x0f 0xf7 - maskmovdqu Vdq, Udq */
9149FNIEMOP_STUB(iemOp_maskmovdqu_Vdq_Udq);
9150/* Opcode 0xf2 0x0f 0xf7 - invalid */
9151
9152/** Opcode 0x0f 0xf8 - psubb Pq, Qq */
9153FNIEMOP_STUB(iemOp_psubb_Pq_Qq);
9154/** Opcode 0x66 0x0f 0xf8 - psubb Vx, W */
9155FNIEMOP_STUB(iemOp_psubb_Vx_W);
9156/* Opcode 0xf2 0x0f 0xf8 - invalid */
9157
9158/** Opcode 0x0f 0xf9 - psubw Pq, Qq */
9159FNIEMOP_STUB(iemOp_psubw_Pq_Qq);
9160/** Opcode 0x66 0x0f 0xf9 - psubw Vx, Wx */
9161FNIEMOP_STUB(iemOp_psubw_Vx_Wx);
9162/* Opcode 0xf2 0x0f 0xf9 - invalid */
9163
9164/** Opcode 0x0f 0xfa - psubd Pq, Qq */
9165FNIEMOP_STUB(iemOp_psubd_Pq_Qq);
9166/** Opcode 0x66 0x0f 0xfa - psubd Vx, Wx */
9167FNIEMOP_STUB(iemOp_psubd_Vx_Wx);
9168/* Opcode 0xf2 0x0f 0xfa - invalid */
9169
9170/** Opcode 0x0f 0xfb - psubq Pq, Qq */
9171FNIEMOP_STUB(iemOp_psubq_Pq_Qq);
9172/** Opcode 0x66 0x0f 0xfb - psubq Vx, W */
9173FNIEMOP_STUB(iemOp_psubq_Vx_W);
9174/* Opcode 0xf2 0x0f 0xfb - invalid */
9175
9176/** Opcode 0x0f 0xfc - paddb Pq, Qq */
9177FNIEMOP_STUB(iemOp_paddb_Pq_Qq);
9178/** Opcode 0x66 0x0f 0xfc - paddb Vx, Wx */
9179FNIEMOP_STUB(iemOp_paddb_Vx_Wx);
9180/* Opcode 0xf2 0x0f 0xfc - invalid */
9181
9182/** Opcode 0x0f 0xfd - paddw Pq, Qq */
9183FNIEMOP_STUB(iemOp_paddw_Pq_Qq);
9184/** Opcode 0x66 0x0f 0xfd - paddw Vx, Wx */
9185FNIEMOP_STUB(iemOp_paddw_Vx_Wx);
9186/* Opcode 0xf2 0x0f 0xfd - invalid */
9187
9188/** Opcode 0x0f 0xfe - paddd Pq, Qq */
9189FNIEMOP_STUB(iemOp_paddd_Pq_Qq);
9190/** Opcode 0x66 0x0f 0xfe - paddd Vx, W */
9191FNIEMOP_STUB(iemOp_paddd_Vx_W);
9192/* Opcode 0xf2 0x0f 0xfe - invalid */
9193
9194
9195/** Opcode **** 0x0f 0xff - UD0 */
9196FNIEMOP_DEF(iemOp_ud0)
9197{
9198 IEMOP_MNEMONIC(ud0, "ud0");
9199 if (pVCpu->iem.s.enmCpuVendor == CPUMCPUVENDOR_INTEL)
9200 {
9201 uint8_t bRm; IEM_OPCODE_GET_NEXT_U8(&bRm); RT_NOREF(bRm);
9202#ifndef TST_IEM_CHECK_MC
9203 RTGCPTR GCPtrEff;
9204 VBOXSTRICTRC rcStrict = iemOpHlpCalcRmEffAddr(pVCpu, bRm, 0, &GCPtrEff);
9205 if (rcStrict != VINF_SUCCESS)
9206 return rcStrict;
9207#endif
9208 IEMOP_HLP_DONE_DECODING();
9209 }
9210 return IEMOP_RAISE_INVALID_OPCODE();
9211}
9212
9213
9214
9215/**
9216 * Two byte opcode map, first byte 0x0f.
9217 *
9218 * @remarks The g_apfnVexMap1 table is currently a subset of this one, so please
9219 * check if it needs updating as well when making changes.
9220 */
9221IEM_STATIC const PFNIEMOP g_apfnTwoByteMap[] =
9222{
9223 /* no prefix, 066h prefix f3h prefix, f2h prefix */
9224 /* 0x00 */ IEMOP_X4(iemOp_Grp6),
9225 /* 0x01 */ IEMOP_X4(iemOp_Grp7),
9226 /* 0x02 */ IEMOP_X4(iemOp_lar_Gv_Ew),
9227 /* 0x03 */ IEMOP_X4(iemOp_lsl_Gv_Ew),
9228 /* 0x04 */ IEMOP_X4(iemOp_Invalid),
9229 /* 0x05 */ IEMOP_X4(iemOp_syscall),
9230 /* 0x06 */ IEMOP_X4(iemOp_clts),
9231 /* 0x07 */ IEMOP_X4(iemOp_sysret),
9232 /* 0x08 */ IEMOP_X4(iemOp_invd),
9233 /* 0x09 */ IEMOP_X4(iemOp_wbinvd),
9234 /* 0x0a */ IEMOP_X4(iemOp_Invalid),
9235 /* 0x0b */ IEMOP_X4(iemOp_ud2),
9236 /* 0x0c */ IEMOP_X4(iemOp_Invalid),
9237 /* 0x0d */ IEMOP_X4(iemOp_nop_Ev_GrpP),
9238 /* 0x0e */ IEMOP_X4(iemOp_femms),
9239 /* 0x0f */ IEMOP_X4(iemOp_3Dnow),
9240
9241 /* 0x10 */ iemOp_movups_Vps_Wps, iemOp_movupd_Vpd_Wpd, iemOp_movss_Vss_Wss, iemOp_movsd_Vsd_Wsd,
9242 /* 0x11 */ iemOp_movups_Wps_Vps, iemOp_movupd_Wpd_Vpd, iemOp_movss_Wss_Vss, iemOp_movsd_Wsd_Vsd,
9243 /* 0x12 */ iemOp_movlps_Vq_Mq__movhlps, iemOp_movlpd_Vq_Mq, iemOp_movsldup_Vdq_Wdq, iemOp_movddup_Vdq_Wdq,
9244 /* 0x13 */ iemOp_movlps_Mq_Vq, iemOp_movlpd_Mq_Vq, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9245 /* 0x14 */ iemOp_unpcklps_Vx_Wx, iemOp_unpcklpd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9246 /* 0x15 */ iemOp_unpckhps_Vx_Wx, iemOp_unpckhpd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9247 /* 0x16 */ iemOp_movhps_Vdq_Mq__movlhps_Vdq_Uq, iemOp_movhpd_Vdq_Mq, iemOp_movshdup_Vdq_Wdq, iemOp_InvalidNeedRM,
9248 /* 0x17 */ iemOp_movhps_Mq_Vq, iemOp_movhpd_Mq_Vq, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9249 /* 0x18 */ IEMOP_X4(iemOp_prefetch_Grp16),
9250 /* 0x19 */ IEMOP_X4(iemOp_nop_Ev),
9251 /* 0x1a */ IEMOP_X4(iemOp_nop_Ev),
9252 /* 0x1b */ IEMOP_X4(iemOp_nop_Ev),
9253 /* 0x1c */ IEMOP_X4(iemOp_nop_Ev),
9254 /* 0x1d */ IEMOP_X4(iemOp_nop_Ev),
9255 /* 0x1e */ IEMOP_X4(iemOp_nop_Ev),
9256 /* 0x1f */ IEMOP_X4(iemOp_nop_Ev),
9257
9258 /* 0x20 */ iemOp_mov_Rd_Cd, iemOp_mov_Rd_Cd, iemOp_mov_Rd_Cd, iemOp_mov_Rd_Cd,
9259 /* 0x21 */ iemOp_mov_Rd_Dd, iemOp_mov_Rd_Dd, iemOp_mov_Rd_Dd, iemOp_mov_Rd_Dd,
9260 /* 0x22 */ iemOp_mov_Cd_Rd, iemOp_mov_Cd_Rd, iemOp_mov_Cd_Rd, iemOp_mov_Cd_Rd,
9261 /* 0x23 */ iemOp_mov_Dd_Rd, iemOp_mov_Dd_Rd, iemOp_mov_Dd_Rd, iemOp_mov_Dd_Rd,
9262 /* 0x24 */ iemOp_mov_Rd_Td, iemOp_mov_Rd_Td, iemOp_mov_Rd_Td, iemOp_mov_Rd_Td,
9263 /* 0x25 */ iemOp_Invalid, iemOp_Invalid, iemOp_Invalid, iemOp_Invalid,
9264 /* 0x26 */ iemOp_mov_Td_Rd, iemOp_mov_Td_Rd, iemOp_mov_Td_Rd, iemOp_mov_Td_Rd,
9265 /* 0x27 */ iemOp_Invalid, iemOp_Invalid, iemOp_Invalid, iemOp_Invalid,
9266 /* 0x28 */ iemOp_movaps_Vps_Wps, iemOp_movapd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9267 /* 0x29 */ iemOp_movaps_Wps_Vps, iemOp_movapd_Wpd_Vpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9268 /* 0x2a */ iemOp_cvtpi2ps_Vps_Qpi, iemOp_cvtpi2pd_Vpd_Qpi, iemOp_cvtsi2ss_Vss_Ey, iemOp_cvtsi2sd_Vsd_Ey,
9269 /* 0x2b */ iemOp_movntps_Mps_Vps, iemOp_movntpd_Mpd_Vpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9270 /* 0x2c */ iemOp_cvttps2pi_Ppi_Wps, iemOp_cvttpd2pi_Ppi_Wpd, iemOp_cvttss2si_Gy_Wss, iemOp_cvttsd2si_Gy_Wsd,
9271 /* 0x2d */ iemOp_cvtps2pi_Ppi_Wps, iemOp_cvtpd2pi_Qpi_Wpd, iemOp_cvtss2si_Gy_Wss, iemOp_cvtsd2si_Gy_Wsd,
9272 /* 0x2e */ iemOp_ucomiss_Vss_Wss, iemOp_ucomisd_Vsd_Wsd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9273 /* 0x2f */ iemOp_comiss_Vss_Wss, iemOp_comisd_Vsd_Wsd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9274
9275 /* 0x30 */ IEMOP_X4(iemOp_wrmsr),
9276 /* 0x31 */ IEMOP_X4(iemOp_rdtsc),
9277 /* 0x32 */ IEMOP_X4(iemOp_rdmsr),
9278 /* 0x33 */ IEMOP_X4(iemOp_rdpmc),
9279 /* 0x34 */ IEMOP_X4(iemOp_sysenter),
9280 /* 0x35 */ IEMOP_X4(iemOp_sysexit),
9281 /* 0x36 */ IEMOP_X4(iemOp_Invalid),
9282 /* 0x37 */ IEMOP_X4(iemOp_getsec),
9283 /* 0x38 */ IEMOP_X4(iemOp_3byte_Esc_0f_38),
9284 /* 0x39 */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRM),
9285 /* 0x3a */ IEMOP_X4(iemOp_3byte_Esc_0f_3a),
9286 /* 0x3b */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRMImm8),
9287 /* 0x3c */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRM),
9288 /* 0x3d */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRM),
9289 /* 0x3e */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRMImm8),
9290 /* 0x3f */ IEMOP_X4(iemOp_InvalidNeed3ByteEscRMImm8),
9291
9292 /* 0x40 */ IEMOP_X4(iemOp_cmovo_Gv_Ev),
9293 /* 0x41 */ IEMOP_X4(iemOp_cmovno_Gv_Ev),
9294 /* 0x42 */ IEMOP_X4(iemOp_cmovc_Gv_Ev),
9295 /* 0x43 */ IEMOP_X4(iemOp_cmovnc_Gv_Ev),
9296 /* 0x44 */ IEMOP_X4(iemOp_cmove_Gv_Ev),
9297 /* 0x45 */ IEMOP_X4(iemOp_cmovne_Gv_Ev),
9298 /* 0x46 */ IEMOP_X4(iemOp_cmovbe_Gv_Ev),
9299 /* 0x47 */ IEMOP_X4(iemOp_cmovnbe_Gv_Ev),
9300 /* 0x48 */ IEMOP_X4(iemOp_cmovs_Gv_Ev),
9301 /* 0x49 */ IEMOP_X4(iemOp_cmovns_Gv_Ev),
9302 /* 0x4a */ IEMOP_X4(iemOp_cmovp_Gv_Ev),
9303 /* 0x4b */ IEMOP_X4(iemOp_cmovnp_Gv_Ev),
9304 /* 0x4c */ IEMOP_X4(iemOp_cmovl_Gv_Ev),
9305 /* 0x4d */ IEMOP_X4(iemOp_cmovnl_Gv_Ev),
9306 /* 0x4e */ IEMOP_X4(iemOp_cmovle_Gv_Ev),
9307 /* 0x4f */ IEMOP_X4(iemOp_cmovnle_Gv_Ev),
9308
9309 /* 0x50 */ iemOp_movmskps_Gy_Ups, iemOp_movmskpd_Gy_Upd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9310 /* 0x51 */ iemOp_sqrtps_Vps_Wps, iemOp_sqrtpd_Vpd_Wpd, iemOp_sqrtss_Vss_Wss, iemOp_sqrtsd_Vsd_Wsd,
9311 /* 0x52 */ iemOp_rsqrtps_Vps_Wps, iemOp_InvalidNeedRM, iemOp_rsqrtss_Vss_Wss, iemOp_InvalidNeedRM,
9312 /* 0x53 */ iemOp_rcpps_Vps_Wps, iemOp_InvalidNeedRM, iemOp_rcpss_Vss_Wss, iemOp_InvalidNeedRM,
9313 /* 0x54 */ iemOp_andps_Vps_Wps, iemOp_andpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9314 /* 0x55 */ iemOp_andnps_Vps_Wps, iemOp_andnpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9315 /* 0x56 */ iemOp_orps_Vps_Wps, iemOp_orpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9316 /* 0x57 */ iemOp_xorps_Vps_Wps, iemOp_xorpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9317 /* 0x58 */ iemOp_addps_Vps_Wps, iemOp_addpd_Vpd_Wpd, iemOp_addss_Vss_Wss, iemOp_addsd_Vsd_Wsd,
9318 /* 0x59 */ iemOp_mulps_Vps_Wps, iemOp_mulpd_Vpd_Wpd, iemOp_mulss_Vss_Wss, iemOp_mulsd_Vsd_Wsd,
9319 /* 0x5a */ iemOp_cvtps2pd_Vpd_Wps, iemOp_cvtpd2ps_Vps_Wpd, iemOp_cvtss2sd_Vsd_Wss, iemOp_cvtsd2ss_Vss_Wsd,
9320 /* 0x5b */ iemOp_cvtdq2ps_Vps_Wdq, iemOp_cvtps2dq_Vdq_Wps, iemOp_cvttps2dq_Vdq_Wps, iemOp_InvalidNeedRM,
9321 /* 0x5c */ iemOp_subps_Vps_Wps, iemOp_subpd_Vpd_Wpd, iemOp_subss_Vss_Wss, iemOp_subsd_Vsd_Wsd,
9322 /* 0x5d */ iemOp_minps_Vps_Wps, iemOp_minpd_Vpd_Wpd, iemOp_minss_Vss_Wss, iemOp_minsd_Vsd_Wsd,
9323 /* 0x5e */ iemOp_divps_Vps_Wps, iemOp_divpd_Vpd_Wpd, iemOp_divss_Vss_Wss, iemOp_divsd_Vsd_Wsd,
9324 /* 0x5f */ iemOp_maxps_Vps_Wps, iemOp_maxpd_Vpd_Wpd, iemOp_maxss_Vss_Wss, iemOp_maxsd_Vsd_Wsd,
9325
9326 /* 0x60 */ iemOp_punpcklbw_Pq_Qd, iemOp_punpcklbw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9327 /* 0x61 */ iemOp_punpcklwd_Pq_Qd, iemOp_punpcklwd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9328 /* 0x62 */ iemOp_punpckldq_Pq_Qd, iemOp_punpckldq_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9329 /* 0x63 */ iemOp_packsswb_Pq_Qq, iemOp_packsswb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9330 /* 0x64 */ iemOp_pcmpgtb_Pq_Qq, iemOp_pcmpgtb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9331 /* 0x65 */ iemOp_pcmpgtw_Pq_Qq, iemOp_pcmpgtw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9332 /* 0x66 */ iemOp_pcmpgtd_Pq_Qq, iemOp_pcmpgtd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9333 /* 0x67 */ iemOp_packuswb_Pq_Qq, iemOp_packuswb_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9334 /* 0x68 */ iemOp_punpckhbw_Pq_Qd, iemOp_punpckhbw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9335 /* 0x69 */ iemOp_punpckhwd_Pq_Qd, iemOp_punpckhwd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9336 /* 0x6a */ iemOp_punpckhdq_Pq_Qd, iemOp_punpckhdq_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9337 /* 0x6b */ iemOp_packssdw_Pq_Qd, iemOp_packssdw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9338 /* 0x6c */ iemOp_InvalidNeedRM, iemOp_punpcklqdq_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9339 /* 0x6d */ iemOp_InvalidNeedRM, iemOp_punpckhqdq_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9340 /* 0x6e */ iemOp_movd_q_Pd_Ey, iemOp_movd_q_Vy_Ey, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9341 /* 0x6f */ iemOp_movq_Pq_Qq, iemOp_movdqa_Vdq_Wdq, iemOp_movdqu_Vdq_Wdq, iemOp_InvalidNeedRM,
9342
9343 /* 0x70 */ iemOp_pshufw_Pq_Qq_Ib, iemOp_pshufd_Vx_Wx_Ib, iemOp_pshufhw_Vx_Wx_Ib, iemOp_pshuflw_Vx_Wx_Ib,
9344 /* 0x71 */ IEMOP_X4(iemOp_Grp12),
9345 /* 0x72 */ IEMOP_X4(iemOp_Grp13),
9346 /* 0x73 */ IEMOP_X4(iemOp_Grp14),
9347 /* 0x74 */ iemOp_pcmpeqb_Pq_Qq, iemOp_pcmpeqb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9348 /* 0x75 */ iemOp_pcmpeqw_Pq_Qq, iemOp_pcmpeqw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9349 /* 0x76 */ iemOp_pcmpeqd_Pq_Qq, iemOp_pcmpeqd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9350 /* 0x77 */ iemOp_emms, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9351
9352 /* 0x78 */ iemOp_vmread_Ey_Gy, iemOp_AmdGrp17, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9353 /* 0x79 */ iemOp_vmwrite_Gy_Ey, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9354 /* 0x7a */ iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9355 /* 0x7b */ iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9356 /* 0x7c */ iemOp_InvalidNeedRM, iemOp_haddpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_haddps_Vps_Wps,
9357 /* 0x7d */ iemOp_InvalidNeedRM, iemOp_hsubpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_hsubps_Vps_Wps,
9358 /* 0x7e */ iemOp_movd_q_Ey_Pd, iemOp_movd_q_Ey_Vy, iemOp_movq_Vq_Wq, iemOp_InvalidNeedRM,
9359 /* 0x7f */ iemOp_movq_Qq_Pq, iemOp_movdqa_Wx_Vx, iemOp_movdqu_Wx_Vx, iemOp_InvalidNeedRM,
9360
9361 /* 0x80 */ IEMOP_X4(iemOp_jo_Jv),
9362 /* 0x81 */ IEMOP_X4(iemOp_jno_Jv),
9363 /* 0x82 */ IEMOP_X4(iemOp_jc_Jv),
9364 /* 0x83 */ IEMOP_X4(iemOp_jnc_Jv),
9365 /* 0x84 */ IEMOP_X4(iemOp_je_Jv),
9366 /* 0x85 */ IEMOP_X4(iemOp_jne_Jv),
9367 /* 0x86 */ IEMOP_X4(iemOp_jbe_Jv),
9368 /* 0x87 */ IEMOP_X4(iemOp_jnbe_Jv),
9369 /* 0x88 */ IEMOP_X4(iemOp_js_Jv),
9370 /* 0x89 */ IEMOP_X4(iemOp_jns_Jv),
9371 /* 0x8a */ IEMOP_X4(iemOp_jp_Jv),
9372 /* 0x8b */ IEMOP_X4(iemOp_jnp_Jv),
9373 /* 0x8c */ IEMOP_X4(iemOp_jl_Jv),
9374 /* 0x8d */ IEMOP_X4(iemOp_jnl_Jv),
9375 /* 0x8e */ IEMOP_X4(iemOp_jle_Jv),
9376 /* 0x8f */ IEMOP_X4(iemOp_jnle_Jv),
9377
9378 /* 0x90 */ IEMOP_X4(iemOp_seto_Eb),
9379 /* 0x91 */ IEMOP_X4(iemOp_setno_Eb),
9380 /* 0x92 */ IEMOP_X4(iemOp_setc_Eb),
9381 /* 0x93 */ IEMOP_X4(iemOp_setnc_Eb),
9382 /* 0x94 */ IEMOP_X4(iemOp_sete_Eb),
9383 /* 0x95 */ IEMOP_X4(iemOp_setne_Eb),
9384 /* 0x96 */ IEMOP_X4(iemOp_setbe_Eb),
9385 /* 0x97 */ IEMOP_X4(iemOp_setnbe_Eb),
9386 /* 0x98 */ IEMOP_X4(iemOp_sets_Eb),
9387 /* 0x99 */ IEMOP_X4(iemOp_setns_Eb),
9388 /* 0x9a */ IEMOP_X4(iemOp_setp_Eb),
9389 /* 0x9b */ IEMOP_X4(iemOp_setnp_Eb),
9390 /* 0x9c */ IEMOP_X4(iemOp_setl_Eb),
9391 /* 0x9d */ IEMOP_X4(iemOp_setnl_Eb),
9392 /* 0x9e */ IEMOP_X4(iemOp_setle_Eb),
9393 /* 0x9f */ IEMOP_X4(iemOp_setnle_Eb),
9394
9395 /* 0xa0 */ IEMOP_X4(iemOp_push_fs),
9396 /* 0xa1 */ IEMOP_X4(iemOp_pop_fs),
9397 /* 0xa2 */ IEMOP_X4(iemOp_cpuid),
9398 /* 0xa3 */ IEMOP_X4(iemOp_bt_Ev_Gv),
9399 /* 0xa4 */ IEMOP_X4(iemOp_shld_Ev_Gv_Ib),
9400 /* 0xa5 */ IEMOP_X4(iemOp_shld_Ev_Gv_CL),
9401 /* 0xa6 */ IEMOP_X4(iemOp_InvalidNeedRM),
9402 /* 0xa7 */ IEMOP_X4(iemOp_InvalidNeedRM),
9403 /* 0xa8 */ IEMOP_X4(iemOp_push_gs),
9404 /* 0xa9 */ IEMOP_X4(iemOp_pop_gs),
9405 /* 0xaa */ IEMOP_X4(iemOp_rsm),
9406 /* 0xab */ IEMOP_X4(iemOp_bts_Ev_Gv),
9407 /* 0xac */ IEMOP_X4(iemOp_shrd_Ev_Gv_Ib),
9408 /* 0xad */ IEMOP_X4(iemOp_shrd_Ev_Gv_CL),
9409 /* 0xae */ IEMOP_X4(iemOp_Grp15),
9410 /* 0xaf */ IEMOP_X4(iemOp_imul_Gv_Ev),
9411
9412 /* 0xb0 */ IEMOP_X4(iemOp_cmpxchg_Eb_Gb),
9413 /* 0xb1 */ IEMOP_X4(iemOp_cmpxchg_Ev_Gv),
9414 /* 0xb2 */ IEMOP_X4(iemOp_lss_Gv_Mp),
9415 /* 0xb3 */ IEMOP_X4(iemOp_btr_Ev_Gv),
9416 /* 0xb4 */ IEMOP_X4(iemOp_lfs_Gv_Mp),
9417 /* 0xb5 */ IEMOP_X4(iemOp_lgs_Gv_Mp),
9418 /* 0xb6 */ IEMOP_X4(iemOp_movzx_Gv_Eb),
9419 /* 0xb7 */ IEMOP_X4(iemOp_movzx_Gv_Ew),
9420 /* 0xb8 */ iemOp_jmpe, iemOp_InvalidNeedRM, iemOp_popcnt_Gv_Ev, iemOp_InvalidNeedRM,
9421 /* 0xb9 */ IEMOP_X4(iemOp_Grp10),
9422 /* 0xba */ IEMOP_X4(iemOp_Grp8),
9423 /* 0xbb */ IEMOP_X4(iemOp_btc_Ev_Gv), // 0xf3?
9424 /* 0xbc */ iemOp_bsf_Gv_Ev, iemOp_bsf_Gv_Ev, iemOp_tzcnt_Gv_Ev, iemOp_bsf_Gv_Ev,
9425 /* 0xbd */ iemOp_bsr_Gv_Ev, iemOp_bsr_Gv_Ev, iemOp_lzcnt_Gv_Ev, iemOp_bsr_Gv_Ev,
9426 /* 0xbe */ IEMOP_X4(iemOp_movsx_Gv_Eb),
9427 /* 0xbf */ IEMOP_X4(iemOp_movsx_Gv_Ew),
9428
9429 /* 0xc0 */ IEMOP_X4(iemOp_xadd_Eb_Gb),
9430 /* 0xc1 */ IEMOP_X4(iemOp_xadd_Ev_Gv),
9431 /* 0xc2 */ iemOp_cmpps_Vps_Wps_Ib, iemOp_cmppd_Vpd_Wpd_Ib, iemOp_cmpss_Vss_Wss_Ib, iemOp_cmpsd_Vsd_Wsd_Ib,
9432 /* 0xc3 */ iemOp_movnti_My_Gy, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9433 /* 0xc4 */ iemOp_pinsrw_Pq_RyMw_Ib, iemOp_pinsrw_Vdq_RyMw_Ib, iemOp_InvalidNeedRMImm8, iemOp_InvalidNeedRMImm8,
9434 /* 0xc5 */ iemOp_pextrw_Gd_Nq_Ib, iemOp_pextrw_Gd_Udq_Ib, iemOp_InvalidNeedRMImm8, iemOp_InvalidNeedRMImm8,
9435 /* 0xc6 */ iemOp_shufps_Vps_Wps_Ib, iemOp_shufpd_Vpd_Wpd_Ib, iemOp_InvalidNeedRMImm8, iemOp_InvalidNeedRMImm8,
9436 /* 0xc7 */ IEMOP_X4(iemOp_Grp9),
9437 /* 0xc8 */ IEMOP_X4(iemOp_bswap_rAX_r8),
9438 /* 0xc9 */ IEMOP_X4(iemOp_bswap_rCX_r9),
9439 /* 0xca */ IEMOP_X4(iemOp_bswap_rDX_r10),
9440 /* 0xcb */ IEMOP_X4(iemOp_bswap_rBX_r11),
9441 /* 0xcc */ IEMOP_X4(iemOp_bswap_rSP_r12),
9442 /* 0xcd */ IEMOP_X4(iemOp_bswap_rBP_r13),
9443 /* 0xce */ IEMOP_X4(iemOp_bswap_rSI_r14),
9444 /* 0xcf */ IEMOP_X4(iemOp_bswap_rDI_r15),
9445
9446 /* 0xd0 */ iemOp_InvalidNeedRM, iemOp_addsubpd_Vpd_Wpd, iemOp_InvalidNeedRM, iemOp_addsubps_Vps_Wps,
9447 /* 0xd1 */ iemOp_psrlw_Pq_Qq, iemOp_psrlw_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9448 /* 0xd2 */ iemOp_psrld_Pq_Qq, iemOp_psrld_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9449 /* 0xd3 */ iemOp_psrlq_Pq_Qq, iemOp_psrlq_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9450 /* 0xd4 */ iemOp_paddq_Pq_Qq, iemOp_paddq_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9451 /* 0xd5 */ iemOp_pmullw_Pq_Qq, iemOp_pmullw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9452 /* 0xd6 */ iemOp_InvalidNeedRM, iemOp_movq_Wq_Vq, iemOp_movq2dq_Vdq_Nq, iemOp_movdq2q_Pq_Uq,
9453 /* 0xd7 */ iemOp_pmovmskb_Gd_Nq, iemOp_pmovmskb_Gd_Ux, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9454 /* 0xd8 */ iemOp_psubusb_Pq_Qq, iemOp_psubusb_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9455 /* 0xd9 */ iemOp_psubusw_Pq_Qq, iemOp_psubusw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9456 /* 0xda */ iemOp_pminub_Pq_Qq, iemOp_pminub_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9457 /* 0xdb */ iemOp_pand_Pq_Qq, iemOp_pand_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9458 /* 0xdc */ iemOp_paddusb_Pq_Qq, iemOp_paddusb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9459 /* 0xdd */ iemOp_paddusw_Pq_Qq, iemOp_paddusw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9460 /* 0xde */ iemOp_pmaxub_Pq_Qq, iemOp_pmaxub_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9461 /* 0xdf */ iemOp_pandn_Pq_Qq, iemOp_pandn_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9462
9463 /* 0xe0 */ iemOp_pavgb_Pq_Qq, iemOp_pavgb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9464 /* 0xe1 */ iemOp_psraw_Pq_Qq, iemOp_psraw_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9465 /* 0xe2 */ iemOp_psrad_Pq_Qq, iemOp_psrad_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9466 /* 0xe3 */ iemOp_pavgw_Pq_Qq, iemOp_pavgw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9467 /* 0xe4 */ iemOp_pmulhuw_Pq_Qq, iemOp_pmulhuw_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9468 /* 0xe5 */ iemOp_pmulhw_Pq_Qq, iemOp_pmulhw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9469 /* 0xe6 */ iemOp_InvalidNeedRM, iemOp_cvttpd2dq_Vx_Wpd, iemOp_cvtdq2pd_Vx_Wpd, iemOp_cvtpd2dq_Vx_Wpd,
9470 /* 0xe7 */ iemOp_movntq_Mq_Pq, iemOp_movntdq_Mdq_Vdq, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9471 /* 0xe8 */ iemOp_psubsb_Pq_Qq, iemOp_psubsb_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9472 /* 0xe9 */ iemOp_psubsw_Pq_Qq, iemOp_psubsw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9473 /* 0xea */ iemOp_pminsw_Pq_Qq, iemOp_pminsw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9474 /* 0xeb */ iemOp_por_Pq_Qq, iemOp_por_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9475 /* 0xec */ iemOp_paddsb_Pq_Qq, iemOp_paddsb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9476 /* 0xed */ iemOp_paddsw_Pq_Qq, iemOp_paddsw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9477 /* 0xee */ iemOp_pmaxsw_Pq_Qq, iemOp_pmaxsw_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9478 /* 0xef */ iemOp_pxor_Pq_Qq, iemOp_pxor_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9479
9480 /* 0xf0 */ iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM, iemOp_lddqu_Vx_Mx,
9481 /* 0xf1 */ iemOp_psllw_Pq_Qq, iemOp_psllw_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9482 /* 0xf2 */ iemOp_pslld_Pq_Qq, iemOp_pslld_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9483 /* 0xf3 */ iemOp_psllq_Pq_Qq, iemOp_psllq_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9484 /* 0xf4 */ iemOp_pmuludq_Pq_Qq, iemOp_pmuludq_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9485 /* 0xf5 */ iemOp_pmaddwd_Pq_Qq, iemOp_pmaddwd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9486 /* 0xf6 */ iemOp_psadbw_Pq_Qq, iemOp_psadbw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9487 /* 0xf7 */ iemOp_maskmovq_Pq_Nq, iemOp_maskmovdqu_Vdq_Udq, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9488 /* 0xf8 */ iemOp_psubb_Pq_Qq, iemOp_psubb_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9489 /* 0xf9 */ iemOp_psubw_Pq_Qq, iemOp_psubw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9490 /* 0xfa */ iemOp_psubd_Pq_Qq, iemOp_psubd_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9491 /* 0xfb */ iemOp_psubq_Pq_Qq, iemOp_psubq_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9492 /* 0xfc */ iemOp_paddb_Pq_Qq, iemOp_paddb_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9493 /* 0xfd */ iemOp_paddw_Pq_Qq, iemOp_paddw_Vx_Wx, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9494 /* 0xfe */ iemOp_paddd_Pq_Qq, iemOp_paddd_Vx_W, iemOp_InvalidNeedRM, iemOp_InvalidNeedRM,
9495 /* 0xff */ IEMOP_X4(iemOp_ud0),
9496};
9497AssertCompile(RT_ELEMENTS(g_apfnTwoByteMap) == 1024);
9498
9499/** @} */
9500
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette