x86 SIMD instruction listings
Part of a series on |
x86 instruction listings |
---|
|
The x86 instruction set has several times been extended with SIMD (Single instruction, multiple data) instruction set extensions. These extensions, starting from the MMX instruction set extension introduced with Pentium MMX in 1997, typically define sets of wide registers and instructions that subdivide these registers into fixed-size lanes and perform a computation for each lane in parallel.
Summary of SIMD extensions
[edit]The main SIMD instruction set extensions that have been introduced for x86 are:
SIMD instruction set extension | Year | Description | Added in |
---|---|---|---|
1997 | A set of 57 integer SIMD instruction acting on 64-bit vectors, mostly providing 8/16/32-bit lane-width operations.
Repurposed the old x87 FPU register-file as a bank of eight 64-bit vector registers, referred to as MM0..MM7 when used for MMX instructions. |
Intel Pentium MMX, AMD K6, Intel Pentium II, Cyrix 6x86MX, MediaGXm, Rise mP6, IDT WinChip C6, Transmeta Crusoe | |
1999 | "Katmai New Instructions" - introduced a set of 70 new instructions. Most but not all of these instructions provide scalar and vector operations on 32-bit floating-point values in 128-bit SIMD vector registers. (Some of the SSE instructions were instead new MMX instructions and non-SIMD instructions such as SFENCE - the subset of SSE that excludes the 128-bit SIMD register instructions is known as "MMX+", and is supported on some AMD processors that didn't implement full SSE, notably early Athlons and Geode LX.)
SSE introduced a new set of eight vector registers XMM0..XMM7, each 128 bits, and a status/control register MXCSR. This set of eight vector registers would later be extended to 16 registers with the introduction of x86-64. |
Intel Pentium III, AMD Athlon XP, VIA C3 "Nehemiah", Transmeta Efficeon | |
|
2000 | Extended SSE with 144 new instructions - mainly additional instructions to work on scalars and vectors of 64-bit floating-point values, as well as 128-bit-vector forms of most of the MMX integer instructions. | Intel Pentium 4, Intel Pentium M, AMD Athlon 64, Transmeta Efficeon, VIA C7 |
|
2004 | "Prescott New Instructions": added a set of 13 new instructions,[a] mostly horizontal add/subtract operations. | Intel Pentium 4 "Prescott", Transmeta Efficeon 8800, Athlon 64 "Venice", VIA C7, Intel Core "Yonah" |
|
2006 | Added a set of 32 new instructions to extend MMX and SSE, including a byte-shuffle instruction. | Intel Core 2 "Merom", VIA Nano 2000, Intel Atom "Bonnell", AMD "Bobcat", AMD FX "Bulldozer" |
|
2007 | AMD-only extension that added a set of 4 instructions, including bitfield insert/extract and scalar non-temporal store instructions. | AMD K10 |
2007 | Added a set of 47 instructions, including variants of integer min/max, widening integer conversions, vector lane insert/extract, and dot-product instructions. | Intel Core 2 "Penryn", VIA Nano 3000, AMD FX "Bulldozer", AMD "Jaguar", Intel Atom "Silvermont" | |
2008 | Added a set of 7 instructions, mostly pertaining to string processing. | Intel Core i7 "Nehalem", AMD FX "Bulldozer", AMD "Jaguar", Intel Atom "Silvermont", VIA Nano QuadCore C4000 | |
2011 | Extended the XMM0..XMM15 vector registers to 256-bit registers, referred to as YMM0..YMM15 when used as full 256-bit registers.
Added three-operand variants of most of the SSE1-4 vector instructions, as well as 256-bit vector variants of most of the SSE1-4 vector instructions acting on 32/64-bit floating-point values. These new instruction variants are all encoded with the new VEX prefix. |
Intel Core i7 "Sandy Bridge", AMD FX "Bulldozer", AMD "Jaguar", VIA Nano QuadCore C4000 | |
2013 | Added three-operand floating-point fused-multiply add operations, scalar and vector variants. | Intel Core i7 "Haswell", AMD FX "Piledriver", Zhaoxin Yongfeng | |
|
2013 | Added 256-bit vector variants of most of the MMX/SSE1-4 vector integer instructions. Also adds vector gather instructions. | Intel Core i7 "Haswell", AMD FX "Excavator", VIA Nano QuadCore C4000 |
2016 | Extended the YMM0..YMM15 vector registers to a set of 32 registers, each 512-bits wide - referred to as ZMM0..ZMM31 when used as 512-bit registers. Also added eight opmask registers K0..K7.
Added 512-bit versions of most of the MMX/SSE/AVX vector instructions, as well as a substantial number of additional instructions. These are mostly encoded with the new EVEX prefix (except for opmask management instructions, which continue to use the VEX prefix.) Added the ability to perform per-vector-lane masking of the operation of most of its vector instructions, by using the opmask registers. Also added embedded rounding controls for floating-point instructions and a scalar-to-vector broadcast function for most instructions that can accept memory operands. |
(See AVX-512#New instructions by sets for additional subsets.) | |
2023 | Added a set of eight new tile registers, referred to as TMM0..TMM7. Each of these tile registers has a size of 8192 bits (16 rows of 64 bytes each). Also added instructions to perform matrix multiplication on these registers with various data formats. | Intel Xeon "Sapphire Rapids" | |
|
2024 | Reformulation of AVX-512 that includes most of the optional AVX-512 subsets as baseline functionality, but also allows for implementations to reduce their maximum supported vector-register width to 256 bits. | Intel Xeon 6 "Granite Rapids" |
|
(2025) | Adds support for rounding modifiers for 256-bit floating-point numbers, as well as a handful of added instructions. | (Intel Diamond Rapids) |
- ^ The count of 13 instructions for SSE3 includes the non-SIMD instructions
MONITOR
andMWAIT
that were also introduced as part of "Prescott New Instructions" - these two instructions are considered to be SSE3 instructions by Intel but not by AMD.
MMX instructions and extended variants thereof
[edit]These instructions are, unless otherwise noted, available in the following forms:
- MMX: 64-bit vectors, operating on mm0..mm7 registers (aliased on top of the old x87 register file)
- SSE2: 128-bit vectors, operating on xmm0..xmm15 registers (xmm0..xmm7 in 32-bit mode)
- AVX: 128-bit vectors, operating on xmm0..xmm15 registers, with a new three-operand encoding enabled by the new VEX prefix. (AVX introduced 256-bit vector registers, but the full width of these vectors was in general not made available for integer SIMD instructions until AVX2.)
- AVX2: 256-bit vectors, operating on ymm0..ymm15 registers (extended versions of the xmm0..xmm15 registers)
- AVX-512: 512-bit vectors, operating on zmm0..zmm31 registers (zmm0..zmm15 are extended versions of the ymm0..ymm15 registers, while zmm16..zmm31 are new to AVX-512). AVX-512 also introduces opmasks, allowing the operation of most instructions to be masked on a per-lane basis by an opmask register (the lane width varies from one instruction to another). AVX-512 also adds broadcast functionality for many of its instructions - this is used with memory source arguments to replicate a single value to all lanes of a vector calculation. The tables below provide indications of whether opmasks and broadcasts are supported for each instruction, and if so, what lane-widths they are using.
For many of the instruction mnemonics, (V)
is used to indicate that the instruction mnemonic exists in forms with and without a leading V - the form with the leading V is used for the VEX/EVEX-prefixed instruction variants introduced by AVX/AVX2/AVX-512, while the form without the leading V is used for legacy MMX/SSE encodings without VEX/EVEX-prefix.
Original Pentium MMX instructions, and SSE2/AVX/AVX-512 extended variants thereof
[edit]Description | Instruction mnemonics | Basic opcode | MMX (no prefix) |
SSE2 (66h prefix) |
AVX (VEX.66 prefix) |
AVX-512 (EVEX.66 prefix) | ||||
---|---|---|---|---|---|---|---|---|---|---|
supported | subset | lane | bcst | |||||||
Empty MMX technology state. (MMX)
Mark all the FP/MMX registers as Empty, so that they can be freely used by later x87 code. |
EMMS (MMX)
|
0F 77
|
EMMS
|
No | VZEROUPPER (L=0)VZEROALL (L=1)[a][b] |
No | — | — | — | |
Zero out upper bits of vector registers YMM0 to YMM15 (AVX) |
VZEROUPPER (AVX)
| |||||||||
Zero out all bits of vector registers YMM0 to YMM15 (AVX) |
VZEROALL (AVX)
| |||||||||
Move scalar value from GPR (general-purpose register) or memory to vector register, with zero-fill | 32-bit | (V)MOVD mm, r/m32 |
0F 6E /r |
Yes | Yes | Yes (L=0,W=0) | Yes (L=0,W=0) | F | No | No |
64-bit (x86-64) |
(V)MOVQ mm, r/m64 ,MOVD mm, r/m64 [c] |
Yes (REX.W) |
Yes (REX.W)[d] |
Yes (L=0,W=1) | Yes (L=0,W=1) | F | No | No | ||
Move scalar value from vector register to GPR or memory | 32-bit | (V)MOVD r/m32, mm |
0F 7E /r |
Yes | Yes | Yes (L=0,W=0) | Yes (L=0,W=0) | F | No | No |
64-bit (x86-64) |
(V)MOVQ r/m64, mm ,MOVD r/m64, mm [c] |
Yes (REX.W) |
Yes (REX.W)[d] |
Yes (L=0,W=1) | Yes (L=0,W=1) | F | No | No | ||
Vector move between vector register and either memory or another vector register.
For move to/from memory, the memory address is required to be aligned for 128-bit VEX-encoded form of |
MOVQ mm/m64, mm (MMX)(V)MOVDQA xmm/m128,xmm
|
0F 7F /r
|
MOVQ
|
MOVDQA
|
VMOVDQA [f]
|
VMOVDQA32 (W0) |
F | 32 | No | |
VMOVDQA64 (W1) |
F | 64 | No | |||||||
MOVQ mm, mm/m64 (MMX)(V)MOVDQA xmm,xmm/m128
|
0F 6F /r
|
MOVQ
|
MOVDQA
|
VMOVDQA [f]
|
VMOVDQA32 (W0) |
F | 32 | No | ||
VMOVDQA64 (W1) |
F | 64 | No | |||||||
Pack 32-bit signed integers to 16-bit, with saturation | (V)PACKSSDW mm, mm/m64 [g] |
0F 6B /r |
Yes | Yes | Yes | Yes (W=0) | BW | 16 | 32 | |
Pack 16-bit signed integers to 8-bit, with saturation | (V)PACKSSWB mm, mm/m64 [g] |
0F 63 /r |
Yes | Yes | Yes | Yes | BW | 8 | No | |
Pack 16-bit unsigned integers to 8-bit, with saturation | (V)PACKUSWB mm, mm/m64 [g] |
0F 67 /r |
Yes | Yes | Yes | Yes | BW | 8 | No | |
Unpack and interleave packed integers from the high halves of two input vectors | 8-bit | (V)PUNPCKHBW mm, mm/m64 [g] |
0F 68 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PUNPCKHWD mm, mm/m64 [g] |
0F 69 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
32-bit | (V)PUNPCKHDQ mm, mm/m64 [g] |
0F 6A /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
Unpack and interleave packed integers from the low halves of two input vectors | 8-bit | (V)PUNPCKLBW mm, mm/m32 [g][h] |
0F 60 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PUNPCKLWD mm, mm/m32 [g][h] |
0F 61 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
32-bit | (V)PUNPCKLDQ mm, mm/m32 [g][h] |
0F 62 /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
Add packed integers | 8-bit | (V)PADDB mm, mm/m64 |
0F FC /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PADDW mm, mm/m64 |
0F FD /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
32-bit | (V)PADDD mm, mm/m64 |
0F FE /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
Add packed signed integers with saturation | 8-bit | (V)PADDSB mm, mm/m64 |
0F EC /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PADDSW mm, mm/m64 |
0F ED /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Add packed unsigned integers with saturation | 8-bit | (V)PADDUSB mm, mm/m64 |
0F DC /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PADDUSW mm, mm/m64 |
0F DD /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Subtract packed integers | 8-bit | (V)PSUBB mm, mm/m64 |
0F F8 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PSUBW mm, mm/m64 |
0F F9 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
32-bit | (V)PSUBD mm, mm/m64 |
0F FA /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
Subtract packed signed integers with saturation | 8-bit | (V)PSUBSB mm, mm/m64 |
0F E8 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PSUBSW mm, mm/m64 |
0F E9 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Subtract packed unsigned integers with saturation | 8-bit | (V)PSUBUSB mm, mm/m64 |
0F D8 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PSUBUSW mm, mm/m64 |
0F D9 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Compare packed integers for equality | 8-bit | (V)PCMPEQB mm, mm/m64 |
0F 74 /r |
Yes | Yes | Yes | Yes[i] | BW | 8 | No |
16-bit | (V)PCMPEQW mm, mm/m64 |
0F 75 /r |
Yes | Yes | Yes | Yes[i] | BW | 16 | No | |
32-bit | (V)PCMPEQD mm, mm/m64 |
0F 76 /r |
Yes | Yes | Yes | Yes (W=0)[i] | F | 32 | 32 | |
Compare packed integers for signed greater-than | 8-bit | (V)PCMPGTB mm, mm/m64 |
0F 64 /r |
Yes | Yes | Yes | Yes[i] | BW | 8 | No |
16-bit | (V)PCMPGTW mm, mm/m64 |
0F 65 /r |
Yes | Yes | Yes | Yes[i] | BW | 16 | No | |
32-bit | (V)PCMPGTD mm, mm/m64 |
0F 66 /r |
Yes | Yes | Yes | Yes (W=0)[i] | F | 32 | 32 | |
Multiply packed 16-bit signed integers, add results pairwise into 32-bit integers | (V)PMADDWD mm, mm/m64 |
0F F5 /r |
Yes | Yes | Yes | Yes[j] | BW | 32 | No | |
Multiply packed 16-bit signed integers, store high 16 bits of results | (V)PMULHW mm, mm/m64 |
0F E5 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Multiply packed 16-bit integers, store low 16 bits of results | (V)PMULLW mm, mm/m64 |
0F D5 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Vector bitwise AND | (V)PAND mm, mm/m64 |
0F DB /r |
Yes | Yes | Yes | VPANDD (W0) |
F | 32 | 32 | |
VPANDQ (W1) |
F | 64 | 64 | |||||||
Vector bitwise AND-NOT | (V)PANDN mm, mm/m64 |
0F DF /r |
Yes | Yes | Yes | VPANDND (W0) |
F | 32 | 32 | |
VPANDNQ (W1) |
F | 64 | 64 | |||||||
Vector bitwise OR | (V)POR mm, mm/m64 |
0F EB /r |
Yes | Yes | Yes | VPORD (W0) |
F | 32 | 32 | |
VPORQ (W1) |
F | 64 | 64 | |||||||
Vector bitwise XOR | (V)PXOR mm, mm/m64 |
0F EE /r |
Yes | Yes | Yes | VPXORD (W0) |
F | 32 | 32 | |
VPXORQ (W1) |
F | 64 | 64 | |||||||
left-shift of packed integers, with common shift-amount | 16-bit | (V)PSLLW mm, imm8 |
0F 71 /6 ib |
Yes | Yes | Yes | Yes | BW | 16 | No |
(V)PSLLW mm, mm/m64 [k] |
0F F1 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | ||
32-bit | (V)PSLLD mm, imm8 |
0F 72 /6 ib |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
(V)PSLLD mm, mm/m64 [k] |
0F F2 /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | No | ||
64-bit | (V)PSLLQ mm, imm8 |
0F 73 /6 ib |
Yes | Yes | Yes | Yes (W=1) | F | 64 | 64 | |
(V)PSLLQ mm, mm/m64 [k] |
0F F3 /r |
Yes | Yes | Yes | Yes (W=1) | F | 64 | No | ||
Right-shift of packed signed integers, with common shift-amount | 16-bit | (V)PSRAW mm, imm8 |
0F 71 /4 ib |
Yes | Yes | Yes | Yes | BW | 16 | No |
(V)PSRAW mm, mm/m64 [k] |
0F E1 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | ||
32-bit | (V)PSRAD mm, imm8 |
0F 72 /4 ib |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
(V)PSRAD mm, mm/m64 [k] |
0F E2 /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | No | ||
Right-shift of packed unsigned integers, with common shift-amount | 16-bit | (V)PSRLW mm, imm8 |
0F 71 /2 ib |
Yes | Yes | Yes | Yes | BW | 16 | No |
(V)PSRLW mm, mm/m64 [k] |
0F D1 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | ||
32-bit | (V)PSRLD mm, imm8 |
0F 72 /2 ib |
Yes | Yes | Yes | Yes (W=0) | F | 32 | 32 | |
(V)PSRLD mm, mm/m64 [k] |
0F D2 /r |
Yes | Yes | Yes | Yes (W=0) | F | 32 | No | ||
64-bit | (V)PSRLQ mm, imm8 |
0F 73 /2 ib |
Yes | Yes | Yes | Yes (W=1) | F | 64 | 64 | |
(V)PSRLQ mm, mm/m64 [k] |
0F D3 /r |
Yes | Yes | Yes | Yes (W=1) | F | 64 | No |
- ^ For code that may potentially mix use of legacy-SSE instructions with AVX instructions, it is strongly recommended to execute a
VZEROUPPER
orVZEROALL
instruction after executing AVX instructions but before executing SSE instructions. If this is not done, any subsequent legacy-SSE code may be subject to severe performance degradation.[1] - ^ On some early AVX implementations (e.g. Sandy Bridge[2]) encoding the
VZEROUPPER
andVZEROALL
instructions with VEX.W=1 will result in #UD - for this reason, it is recommended to encode these instructions with VEX.W=0. - ^ a b The 64-bit move instruction forms that are encoded by using a
REX.W
prefix with the0F 6E
and0F 7E
opcodes are listed with different mnemonics in Intel and AMD documentation —MOVQ
in Intel documentation[3] andMOVD
in AMD documentation.[4]
This is a documentation difference only — the operation performed by these opcodes is the same for Intel and AMD.
This documentation difference applies only to the MMX/SSE forms of these opcodes — for VEX/EVEX-encoded forms, both Intel and AMD use the mnemonicVMOVQ
.) - ^ a b The REX.W-encoded variants of
MOVQ
are available in 64-bit "long mode" only. For SSE2 and later,MOVQ
to and from xmm/ymm/zmm registers can also be encoded withF3 0F 7E /r
and66 0F D6 /r
respectively - these encodings are shorter and available outside 64-bit mode. - ^ On all Intel,[5] AMD[6] and Zhaoxin[7] processors that support AVX, the 128-bit forms of
VMOVDQA
(encoded with a VEX prefix and VEX.L=0) are, when used with a memory argument addressing WB (write-back cacheable) memory, architecturally guaranteed to perform the 128-bit memory access atomically - this applies to both load and store.(Intel and AMD provide somewhat wider guarantees covering more 128-bit instruction variants, but Zhaoxin provides the guarantee for cacheable
VMOVDQA
only.)On processors that support SSE but don't support AVX, the 128-bit forms of SSE load/store instructions such as
MOVAPS
/MOVAPD
/MOVDQA
are not guaranteed to execute atomically — examples of processors where such instructions have been observed to execute non-atomically include Intel Core Duo and AMD K10.[8] - ^ a b
VMOVDQA
is available with a vector length of 256 bits under AVX, not requiring AVX2. - ^ a b c d e f g h i For the
VPACK*
andVPUNPCK*
instructions, encodings with a vector-length wider than 128 bits are available under AVX2 and AVX-512, but the operation of such encodings is split into 128-bit lanes where each 128-bit lane internally performs the same operation as the 128-bit variant of the instruction. - ^ a b c For the memory argument forms of
(V)PUNPCKL*
instructions, the memory argument is half-width only for the MMX variants of the instructions. For SSE/AVX/AVX-512 variants, the width of the memory argument is the full vector width even though only half of it is actually used. - ^ a b c d e f The EVEX-encoded variants of the
VPCMPEQ*
andVPCMPGT*
instructions write their results to AVX-512 opmask registers. This differs from the older non-EVEX variants, which write comparison results as vectors of all-0s/all-1s values to the regular mm/xmm/ymm vector registers. - ^ The
(V)PMADDWD
instruction will add multiplication results pairwise, but will not add the sum to an accumulator. AVX512_VNNI provides the instructionsVDPWSSD
andWDPWSSDS
, which will add multiplication results pairwise, and then also add them to a per-32-bit-lane accumulator. - ^ a b c d e f g h For the MMX packed shift instructions
PSLL*
andPSR*
with a shift-argument taken from a vector source (mm or m64), the shift-amount is considered to be a single 64-bit scalar value - the same shift-amount is used for all lanes of the destination vector. This shift-amount is unsigned and is not masked - all bits are considered (e.g. a shift-amount of0x80000000_00000000
can be specified and will have the same effect as a shift-amount of 64).For all SSE2/AVX/AVX512 extended variants of these instructions, the shift-amount vector argument is considered to be a 128-bit (xmm or m128) argument - the bottom 64 bits are used as the shift-amount.
Packed shift-instructions that can take a variable per-lane shift-amount were introduced in AVX2 for 32/64-bit lanes and AVX512BW for 16-bit lanes (
VPSLLV*
,VPSRLV*
,VPSRAV*
instructions).
MMX instructions added with MMX+/SSE/SSE2/SSSE3, and SSE2/AVX/AVX-512 extended variants thereof
[edit]Description | Instruction mnemonics | Basic opcode | MMX (no prefix) |
SSE2 (66h prefix) |
AVX (VEX.66 prefix) |
AVX-512 (EVEX.66 prefix) | ||||
---|---|---|---|---|---|---|---|---|---|---|
supported | subset | lane | bcst | |||||||
Added with SSE and MMX+ | ||||||||||
Perform shuffle of four 16-bit integers in 64-bit vector (MMX)[a] | PSHUFW mm,mm/m64,imm8 (MMX) |
0F 70 /r ib |
PSHUFW |
PSHUFD |
VPSHUFD |
VPSHUFD (W=0) |
F | 32 | 32 | |
Perform shuffle of four 32-bit integers in 128-bit vector (SSE2) | (V)PSHUFD xmm,xmm/m128,imm8 [b]
| |||||||||
Insert integer into 16-bit vector register lane | (V)PINSRW mm,r32/m16,imm8 |
0F C4 /r ib |
Yes | Yes | Yes (L=0,W=0[c]) | Yes (L=0) | BW | No | No | |
Extract integer from 16-bit vector register lane, with zero-extension | (V)PEXTRW r32,mm,imm8 [d] |
0F C5 /r ib |
Yes | Yes | Yes (L=0,W=0[c]) | Yes (L=0) | BW | No | No | |
Create a bitmask made from the top bit of each byte in the source vector, and store to integer register | (V)PMOVMSKB r32,mm |
0F D7 /r |
Yes | Yes | Yes | No[e] | — | — | — | |
Minimum-value of packed unsigned 8-bit integers | (V)PMINUB mm,mm/m64 |
0F DA /r |
Yes | Yes | Yes | Yes | BW | 8 | No | |
Maximum-value of packed unsigned 8-bit integers | (V)PMAXUB mm,mm/m64 |
0F DE /r |
Yes | Yes | Yes | Yes | BW | 8 | No | |
Minimum-value of packed signed 16-bit integers | (V)PMINSW mm,mm/m64 |
0F EA /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Maximum-value of packed signed 16-bit integers | (V)PMAXSW mm,mm/m64 |
0F EE /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Rounded average of packed unsigned integers. The per-lane operation is:dst ← (src1 + src2 + 1)>>1
|
8-bit | (V)PAVGB mm,mm/m64 |
0F E0 /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PAVGW mm,mm/m64 |
0F E3 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Multiply packed 16-bit unsigned integers, store high 16 bits of results | (V)PMULHUW mm,mm/mm64 |
0F E4 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Store vector register to memory using Non-Temporal Hint. Memory operand required to be aligned for all |
MOVNTQ m64,mm (MMX)(V)MOVNTDQ m128,xmm |
0F E7 /r |
MOVNTQ |
MOVNTDQ |
VMOVNTDQ [f] |
VMOVNTDQ (W=0) |
F | No | No | |
Compute sum of absolute differences for eight 8-bit unsigned integers, storing the result as a 64-bit integer. For vector widths wider than 64 bits (SSE/AVX/AVX-512), this calculation is done separately for each 64-bit lane of the vectors, producing a vector of 64-bit integers. |
(V)PSADBW mm,mm/m64 |
0F F6 /r |
Yes | Yes | Yes | Yes | BW | No | No | |
Unaligned store vector register to memory using byte write-mask, with Non-Temporal Hint. First argument provides data to store, second argument provides byte write-mask (top bit of each byte).[g] Address to store to is given by DS:DI/EDI/RDI (DS: segment overridable with segment-prefix). |
MASKMOVQ mm,mm (MMX)(V)MASKMOVDQU xmm,xmm |
0F F7 /r |
MASKMOVQ |
MASKMOVDQU |
VMASKMOVDQU (L=0)[h] |
No[i] | — | — | — | |
Added with SSE2 | ||||||||||
Multiply packed 32-bit unsigned integers, store full 64-bit result. The input integers are taken from the low 32 bits of each 64-bit vector lane. |
(V)PMULUDQ mm,mm/m64 |
0F F4 /r |
Yes | Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Add packed 64-bit integers | (V)PADDQ mm, mm/m64 |
0F D4 /r |
Yes | Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Subtract packed 64-bit integers | (V)PSUBQ mm,mm/m64 |
0F FB /r |
Yes | Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Added with SSSE3 | ||||||||||
Vector Byte Shuffle | (V)PSHUFB mm,mm/m64 [b] |
0F38 00 /r |
Yes | Yes[j] | Yes | Yes | BW | 8 | No | |
Pairwise horizontal add of packed integers | 16-bit | (V)PHADDW mm,mm/mm64 [b] |
0F38 01 /r |
Yes | Yes | Yes | No | — | — | — |
32-bit | (V)PHADDD mm,mm/mm64 [b] |
0F38 02 /r |
Yes | Yes | Yes | No | — | — | — | |
Pairwise horizontal add of packed 16-bit signed integers, with saturation | (V)PHADDSW mm,mm/mm64 [b] |
0F38 03 /r |
Yes | Yes | Yes | No | — | — | — | |
Multiply packed 8-bit signed and unsigned integers, add results pairwise into 16-bit signed integers with saturation. First operand is treated as unsigned, second operand as signed. | (V)PMADDUBSW mm,mm/m64 |
0F38 04 /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Pairwise horizontal subtract of packed integers. The higher-order integer of each pair is subtracted from the lower-order integer. |
16-bit | (V)PHSUBW mm,mm/m64 [b] |
0F38 05 /r |
Yes | Yes | Yes | No | — | — | — |
32-bit | (V)PHSUBD mm,mm/m64 [b] |
0F38 06 /r |
Yes | Yes | Yes | No | — | — | — | |
Pairwise horizontal subtract of packed 16-bit signed integers, with saturation | (V)PHSUBSW mm,mm/m64 [b] |
0F38 07 /r |
Yes | Yes | Yes | No | — | — | — | |
Modify packed integers in first source argument based on the sign of packed signed integers in second source argument. The per-lane operation performed is:if( src2 < 0 ) dst ← -src1 else if( src2 == 0 ) dst ← 0 else dst ← src1 |
8-bit | (V)PSIGNB mm,mm/m64 |
0F38 08 /r |
Yes | Yes | Yes | No | — | — | — |
16-bit | (V)PSIGNW mm,mm/m64 |
0F38 09 /r |
Yes | Yes | Yes | No | — | — | — | |
32-bit | (V)PSIGND mm,mm/m64 |
0F38 0A /r |
Yes | Yes | Yes | No | — | — | — | |
Multiply packed 16-bit signed integers, then perform rounding and scaling to produce a 16-bit signed integer result. The calculation performed per 16-bit lane is: |
(V)PMULHRSW mm,mm/m64 |
0F38 0B /r |
Yes | Yes | Yes | Yes | BW | 16 | No | |
Absolute value of packed signed integers | 8-bit | (V)PABSB mm,mm/m64 |
0F38 1C /r |
Yes | Yes | Yes | Yes | BW | 8 | No |
16-bit | (V)PABSW mm,mm/m64 |
0F38 1D /r |
Yes | Yes | Yes | Yes | BW | 8 | No | |
32-bit | (V)PABSD mm,mm/m64
|
0F38 1E /r
|
PABSD
|
PABSD
|
VPABSD
|
VPABSD (W0) |
F | 32 | 32 | |
64-bit | VPABSQ xmm,xmm/m128 (AVX-512)
|
VPABSQ (W1) |
F | 64 | 64 | |||||
Packed Align Right. Concatenate two input vectors into a double-size vector, then right-shift by the number of bytes specified by the imm8 argument. The shift-amount is not masked - if the shift-amount is greater than the input vector size, zeroes will be shifted in. |
(V)PALIGNR mm,mm/mm64,imm8 [b] |
0F3A 0F /r ib |
Yes | Yes | Yes | Yes[k] | BW | 8 | No |
- ^ For shuffle of four 16-bit integers in a 64-bit section of a 128-bit XMM register, the SSE2 instructions
PSHUFLW
(opcodeF2 0F 70 /r
) orPSHUFHW
(opcodeF3 0F 70 /r
) may be used. - ^ a b c d e f g h i For the
VPSHUFD
,VPSHUFB
,VPHADD*
,VPHSUB*
andVPALIGNR
instructions, encodings with a vector-length wider than 128 bits are available under AVX2 and/or AVX-512, but the operation of such encodings is split into 128-bit lanes where each 128-bit lane internally performs the same operation as the 128-bit variant of the instruction. - ^ a b For the VEX-encoded forms of the
VPINSRW
andVPEXTRW
instruction, the Intel SDM (as of rev 084) indicates that the instructions must be encoded with VEX.W=0, however neither Intel XED nor AMD APM indicate any such requirement. - ^ The
0F C5 /r ib
variant ofPEXTRW
allows register destination only. For SSE4.1 and later, a variant that allows a memory destination is available with the opcode66 0F 3A 15 /r ib
. - ^ EVEX-prefixed opcode not available. Under AVX-512, a bitmask made from the top bit of each byte can instead be constructed with the
VPMOVB2M
instruction, with opcodeEVEX.F3.0F38.W0 29 /r
, which will store such a bitmask to an opmask register. - ^
VMOVNTDQ
is available with a vector length of 256 bits under AVX, not requiring AVX2. - ^ For the
MASKMOVQ
and(V)MASKMOVDQU
instructions, exception and trap behavior for disabled lanes is implementation-dependent. For example, a given implementation may signal a data breakpoint or a page fault for bytes that are zero-masked and not actually written. - ^ For AVX, masked stores to memory are also available using the
VMASKMOVPS
instruction with opcodeVEX.66.0F38 2E /r
- unlikeVMASKMOVDQU
, this instruction allows 256-bit stores without temporal hints, although its mask is coarser - 4 bytes vs 1 byte per lane. - ^ Opcode not available under AVX-512. Under AVX-512, unaligned masked stores to memory (albeit without temporal hints) can be done with the
VMOVDQU(8|16|32|64)
instructions with opcodeEVEX.F2/F3.0F 7F /r
, using an opmask register to provide a write mask. - ^ For AVX2 and AVX-512 with vectors wider than 128 bits, the
VPSHUFB
instruction is restricted to byte-shuffle within each 128-bit lane. Instructions that can do shuffles across 128-bit lanes include e.g. AVX2'sVPERMD
(shuffle of 32-bit lanes across 256-bit YMM register) and AVX512_VBMI'sVPERMB
(full byte shuffle across 64-byte ZMM register). - ^ For AVX-512,
VPALIGNR
is supported but will perform its operation within each 128-bit lane. For packed alignment shifts that can shift data across 128-bit lanes, AVX512F'sVALIGND
instruction may be used, although its shift-amount is specified in units of 32-bits rather than bytes.
SSE instructions and extended variants thereof
[edit]Regularly-encoded floating-point SSE/SSE2 instructions, and AVX/AVX-512 extended variants thereof
[edit]For the instructions in the below table, the following considerations apply unless otherwise noted:
- Packed instructions are available at all vector lengths (128-bit for SSE2, 128/256-bit for AVX, 128/256/512-bit for AVX-512)
- FP32 variants of instructions are introduced as part of SSE. FP64 variants of instructions are introduced as part of SSE2.
- The AVX-512 variants of the FP32 and FP64 instructions are introduced as part of the AVX512F subset.
- For AVX-512 variants of the instructions, opmasks and broadcasts are available with a width of 32 bits for FP32 operations and 64 bits for FP64 operations. (Broadcasts are available for vector operations only.)
From SSE2 onwards, some data movement/bitwise instructions exist in three forms: an integer form, an FP32 form and an FP64 form. Such instructions are functionally identical, however some processors with SSE2 will implement integer, FP32 and FP64 execution units as three different execution clusters, where forwarding of results from one cluster to another may come with performance penalties and where such penalties can be minimzed by choosing instruction forms appropriately. (For example, there exists three forms of vector bitwise XOR instructions under SSE2 - PXOR
, XORPS
, and XORPD
- these are intended for use on integer, FP32, and FP64 data, respectively.)
Instruction Description | Basic opcode | Single Precision (FP32) | Double Precision (FP64) | AVX-512: RC/SAE
| |||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
Packed (no prefix) | Scalar (F3h prefix) | Packed (66h prefix) | Scalar (F2h prefix) | ||||||||||||||||
SSE instruction | AVX (VEX) |
AVX-512 (EVEX) |
SSE instruction | AVX (VEX) |
AVX-512 (EVEX) |
SSE2 instruction | AVX (VEX) |
AVX-512 (EVEX) |
SSE2 instruction | AVX (VEX) |
AVX-512 (EVEX) | ||||||||
Unaligned load from memory or vector register | 0F 10 /r
|
MOVUPS x,x/m128 |
Yes | Yes[b] | MOVSS x,x/m32 |
Yes | Yes | MOVUPD x,x/m128 |
Yes | Yes[b] | MOVSD x,x/m64 [c] |
Yes | Yes | No | |||||
Unaligned store to memory or vector register | 0F 11 /r
|
MOVUPS x/m128,x |
Yes | Yes[b] | MOVSS x/m32,x |
Yes | Yes | MOVUPD x/m128,x |
Yes | Yes[b] | MOVSD x/m64,x [c] |
Yes | Yes | No | |||||
Load 64 bits from memory or upper half of XMM register into the lower half of XMM register while keeping the upper half unchanged | 0F 12 /r
|
MOVHLPS x,x |
(L0)[d] | (L0)[d] | (MOVSLDUP)[e] | MOVLPD x,m64 |
(L0)[d] | (L0)[d] | (MOVDDUP)[e] | No | |||||||||
MOVLPS x,m64 |
(L0)[d] | (L0)[d] | |||||||||||||||||
Store 64 bits to memory from lower half of XMM register | 0F 13 /r
|
MOVLPS m64,x |
(L0)[d] | (L0)[d] | No | No | No | MOVLPD m64,x |
(L0)[d] | (L0)[d] | No | No | No | No | |||||
Unpack and interleave low-order floating-point values | 0F 14 /r
|
UNPCKLPS x,x/m128
|
Yes[f] | Yes[f] | No | No | No | UNPCKLPD x,x/m128 |
Yes[f] | Yes[f] | No | No | No | No | |||||
Unpack and interleave high-order floating-point values | 0F 15 /r
|
UNPCKHPS x,x/m128 |
Yes[f] | Yes[f] | No | No | No | UNPCKHPD x,x/m128 |
Yes[f] | Yes[f] | No | No | No | No | |||||
Load 64 bits from memory or lower half of XMM register into the upper half of XMM register while keeping the lower half unchanged | 0F 16 /r
|
MOVLHPS x,x |
(L0)[d] | (L0)[d] | (MOVSHDUP)[e] | MOVHPD x,m64 |
(L0)[d] | (L0)[d] | No | No | No | No | |||||||
MOVHPS x,m64 |
(L0)[d] | (L0)[d] | |||||||||||||||||
Store 64 bits to memory from upper half of XMM register | 0F 17 /r
|
MOVHPS m64,x |
(L0)[d] | (L0)[d] | No | No | No | MOVHPD m64,x |
(L0)[d] | (L0)[d] | No | No | No | No | |||||
Aligned load from memory or vector register | 0F 28 /r
|
MOVAPS x,x/m128 |
Yes | Yes[b] | No | No | No | MOVAPD x,x/m128 |
Yes | Yes[b] | No | No | No | No | |||||
Aligned store to memory or vector register | 0F 29 /r
|
MOVAPS x/m128,x |
Yes | Yes[b] | No | No | No | MOVAPD x/m128,x |
Yes | Yes[b] | No | No | No | No | |||||
Integer to floating-point conversion using general-registers, MMX-registers or memory as source | 0F 2A /r
|
CVTPI2PS x,mm/m64 [g] |
No | No | CVTSI2SS x,r/m32 CVTSI2SS x,r/m64 [h] |
Yes | Yes[i] | CVTPI2PD x,mm/m64 [g] |
No | No | CVTSI2SD x,r/m32 CVTSI2SD x,r/m64 [h] |
Yes | Yes[i] | RC | |||||
Non-temporal store to memory from vector register. The packed variants require aligned memory addresses even in VEX/EVEX-encoded forms. |
0F 2B /r
|
MOVNTPS m128,x |
Yes | Yes[i] | MOVNTSS m32,x (AMD SSE4a) |
No | No | MOVNTPD m128,x |
Yes | Yes[i] | MOVNTSD m64,x (AMD SSE4a) |
No | No | No | |||||
Floating-point to integer conversion with truncation, using general-purpose registers or MMX-registers as destination | 0F 2C /r
|
CVTTPS2PI mm,x/m64 |
No | No | CVTTSS2SI r32,x/m32 CVTTSS2SI r64,x/m32 |
Yes | Yes[i] | CVTTPD2PI mm,x/m64 |
No | No | CVTTSD2SI r32,x/m64 CVTTSD2SI r64,x/m64 |
Yes | Yes[i] | SAE | |||||
Floating-point to integer conversion, using general-purpose registers or MMX-registers as destination | 0F 2D /r
|
CVTPS2PI mm,x/m64 [j] |
No | No | CVTSS2SI r32,x/m32 CVTSS2SI r64,x/m32 |
Yes | Yes[i] | CVTPD2PI mm,x/m64 [j] |
No | No | CVTSD2SI r32,x/m64 CVTSD2SI r64,x/m64 |
Yes | Yes[i] | RC | |||||
Unordered compare floating-point values and set EFLAGS. Compares the bottom lanes of xmm vector registers. |
0F 2E /r
|
UCOMISS x,x/m32 |
Yes[a] | Yes[i] | No | No | No | UCOMISD x,x/m64 |
Yes[a] | Yes[i] | No | No | No | SAE | |||||
Compare floating-point values and set EFLAGS. Compares the bottom lanes of xmm vector registers. |
0F 2F /r
|
COMISS x,x/m32 |
Yes[a] | Yes[i] | No | No | No | COMISD x,x/m64 |
Yes[a] | Yes[i] | No | No | No | SAE | |||||
Extract packed floating-point sign mask | 0F 50 /r
|
MOVMSKPS r32,x |
Yes | No[l] | No | No | No | MOVMSKPD r32,x |
Yes | No[l] | No | No | No | — | |||||
Floating-point Square Root | 0F 51 /r
|
SQRTPS x,x/m128 |
Yes | Yes | SQRTSS x,x/m32 |
Yes | Yes | SQRTPD x,x/m128 |
Yes | Yes | SQRTSD x,x/m64 |
Yes | Yes | RC | |||||
Reciprocal Square Root Approximation[m] | 0F 52 /r
|
RSQRTPS x,x/m128 |
Yes | No[n] | RSQRTSS x,x/m32 |
Yes | No[n] | No | No | No[n] | No | No | No[n] | — | |||||
Reciprocal Approximation[m] | 0F 53 /r
|
RCPPS x,x/m128 |
Yes | No[o] | RCPSS x,x/m32 |
Yes | No[o] | No | No | No[o] | No | No | No[o] | — | |||||
Vector bitwise AND | 0F 54 /r
|
ANDPS x,x/m128 |
Yes | (DQ)[p] | No | No | No | ANDPD x,x/m128 |
Yes | (DQ)[p] | No | No | No | No | |||||
Vector bitwise AND-NOT | 0F 55 /r
|
ANDNPS x,x/m128 |
Yes | (DQ)[p] | No | No | No | ANDNPD x,x/m128 |
Yes | (DQ)[p] | No | No | No | No | |||||
Vector bitwise OR | 0F 56 /r
|
ORPS x,x/m128 |
Yes | (DQ)[p] | No | No | No | ORPD x,x/m128 |
Yes | (DQ)[p] | No | No | No | No | |||||
Vector bitwise XOR[q] | 0F 57 /r
|
XORPS x,x/m128 |
Yes | (DQ)[p] | No | No | No | XORPD x,x/m128 |
Yes | (DQ)[p] | No | No | No | No | |||||
Floating-point Add | 0F 58 /r
|
ADDPS x,x/m128 |
Yes | Yes | ADDSS x,x/m32 |
Yes | Yes | ADDPD x,x/m128 |
Yes | Yes | ADDSD x,x/m64 |
Yes | Yes | RC | |||||
Floating-point Multiply | 0F 59 /r
|
MULPS x,x/m128 |
Yes | Yes | MULSS x,x/m32 |
Yes | Yes | MULPD x,x/m128 |
Yes | Yes | MULSD x,x/m64 |
Yes | Yes | RC | |||||
Convert between floating-point formats (FP32→FP64, FP64→FP32) |
0F 5A /r
|
CVTPS2PD x,x/m64 (SSE2) |
Yes | Yes[r] | CVTSS2SD x,x/m32 (SSE2) |
Yes | Yes[r] | CVTPD2PS x,x/m128 |
Yes | Yes[r] | CVTSD2SS x,x/m64 |
Yes | Yes[r] | SAE, RC[s] | |||||
Floating-point Subtract | 0F 5C /r
|
SUBPS x,x/m128 |
Yes | Yes | SUBSS x,x/m32 |
Yes | Yes | SUBPD x,x/m128 |
Yes | Yes | SUBSD x,x/m64 |
Yes | Yes | RC | |||||
Floating-point Minimum Value[t] | 0F 5D /r
|
MINPS x,x/m128 |
Yes | Yes | MINSS x,x/m32 |
Yes | Yes | MINPD x,x/m128 |
Yes | Yes | MINSD x,x/m64 |
Yes | Yes | SAE | |||||
Floating-point Divide | 0F 5E /r
|
DIVPS x,x/m128 |
Yes | Yes | DIVSS x,x/m32 |
Yes | Yes | DIVPD x,x/m128 |
Yes | Yes | DIVSD x,x/m64 |
Yes | Yes | RC | |||||
Floating-point Maximum Value[t] | 0F 5F /r
|
MAXPS x,x/m128 |
Yes | Yes | MAXSS x,x/m32 |
Yes | Yes | MAXPD x,x/m128 |
Yes | Yes | MAXSD x,x/m64 |
Yes | Yes | SAE | |||||
Floating-point compare. Result is written as all-0s/all-1s values (all-1s for comparison true) to vector registers for SSE/AVX, but opmask register for AVX-512. Comparison function is specified by imm8 argument.[u] | 0F C2 /r ib
|
CMPPS x,x/m128,imm8 |
Yes | Yes | CMPSS x,x/m32,imm8 |
Yes | Yes | CMPPD x,x/m128,imm8 |
Yes | Yes | CMPSD x,x/m64,imm8 [c] |
Yes | Yes | SAE | |||||
Packed Interleaved Shuffle. Performs a shuffle on each of its two input arguments, then keeps the bottom half of the shuffle result from its first argument and the top half of the shuffle result from its second argument. |
0F C6 /r ib
|
SHUFPS x,x/m128,imm8 |
Yes | Yes | No | No | No | SHUFPD x,x/m128,imm8 |
Yes | Yes | No | No | No | No |
- ^ a b c d e f The VEX-prefix-encoded variants of the scalar instructions listed in this table should be encoded with VEX.L=0. Setting VEX.L=1 for any of these instructions is allowed but will result in what the Intel SDM describes as "unpredictable behavior across different processor generations". This also applies to VEX-encoded variants of
V(U)COMISS
andV(U)COMISD
. (This behavior does not apply to scalar instructions outside this table, such as e.g.VMOVD
/VMOVQ
, where VEX.L=1 results in an #UD exception.) - ^ a b c d e f g h EVEX-encoded variants of
VMOVAPS
,VMOVUPS
,VMOVAPD
andVMOVUPD
support opmasks but do not support broadcast. - ^ a b c The SSE2
MOVSD
(MOVe Scalar Double-precision) andCMPSD
(CoMPare Scalar Double-precision) instructions have the same names as the older i386MOVSD
(MOVe String Doubleword) andCMPSD
(CoMPare String Doubleword) instructions, however their operations are completely unrelated.At the assembly language level, they can be distinguished by their use of XMM register operands.
- ^ a b c d e f g h i j k l m n o p q r s t For variants of
VMOVLPS
,VMOVHPS
,VMOVLPD
,VMOVHPD
,VMOVLHPS
,VMOVHLPS
encoded with VEX or EVEX prefixes, the only supported vector length is 128 bits (VEX.L=0 or EVEX.L=0).For the EVEX-encoded variants, broadcasts and opmasks are not supported.
- ^ a b c The
MOVSLDUP
,MOVSHDUP
andMOVDDUP
instructions are not regularly-encoded scalar SSE1/2 instructions, but instead irregularly-assigned SSE3 vector instructions. For a description of these instructions, see table below. - ^ a b c d e f g h i j For the
VUNPCK*
,VSHUFPS
andVSHUFPD
instructions, encodings with a vector-length wider than 128 bits are available under AVX2 and AVX-512, but the operation of such encodings is split into 128-bit lanes where each 128-bit lane internally performs the same operation as the 128-bit variant of the instruction (except that forVSHUFPD
, each 128-bit lane will use a different 2-bit part of the instruction's imm8 argument). - ^ a b The
CVTPI2PS
andCVTPI2PD
instructions take their input data as a vector of two 32-bit signed integers from either memory or MMX register. They will cause an x87→MMX transition even if the source operand is a memory operand.For vector int→FP conversions that can accept an xmm/ymm/zmm register or vectors wider than 64 bits as input arguments, SSE2 provides the following irregularly-assigned instructions (see table below):
CVTDQ2PS
(0F 5B /r
)CVTDQ2PD
(F3 0F E6 /r
)
- ^ a b For the
(V)CVTSI2SS
and(V)CVTSI2SD
instructions, variants with a 64-bit source argument are only available in 64-bit long mode and require REX.W, VEX.W or EVEX.W to be set to 1.In 32-bit mode, their source argument is always 32-bit even if VEX.W or EVEX.W is set to 1.
- ^ a b c d e f g h i j k l EVEX-encoded variants of
VMOVNTPS
,VMOVNTSS
VCOMISS
,VCOMISD
,VUCOMISS
,VUCOMISD
VCVTSI2SS
,VCTSI2SD
VCVT(T)SS2SI
,VCVT(T)SD2SI
- ^ a b c d The
CVT(T)PS2PI
andCVT(T)PD2PI
instructions write their result to MMX register as a vector of two 32-bit signed integers.For vector FP→int conversions that can write results to xmm/ymm/zmm registers, SSE2 provides the following irregularly-assigned instructions (see table below):
CVTPS2DQ
(66 0F 5B /r
)CVTTPS2DQ
(F3 0F 5B /r
)CVTPD2DQ
(F2 0F E6 /r
)CVTTPD2DQ
(66 0F E6 /r
)
- ^ a b c d For the
(V)CVT(T)SS2SI
and(V)CVT(T)SD2SI
instructions, variants with a 64-bit destination register are only available in 64-bit long mode and require REX.W, VEX.W or EVEX.W to be set to 1.In 32-bit mode, their destination register is always 32-bit even if VEX.W or EVEX.W is set to 1.
- ^ a b This instruction cannot be EVEX-encoded. Under AVX512DQ, extracting packed floating-point sign-bits can instead be done with the
VPMOVD2M
andVPMOVQ2M
instructions. - ^ a b The
(V)RCPSS
,(V)RCPPS
,(V)RSQRTSS
and(V)RSQRTPS
approximation instructions compute their result with a relative error of at most . The exact calculation is implementation-specific and known to vary between different x86 CPUs. - ^ a b c d This instruction cannot be EVEX-encoded. Instead, AVX512F provides different opcodes -
EVEX.66.0F38 4E/4F /r
- for its newVRSQRT14*
reciprocal square root approximation instructions.The main difference between the AVX-512
VRSQRT14*
instructions and the older SSE/AVX(V)RSQRT*
instructions is that the AVX-512VRSQRT14*
instructions have their operation defined in a bit-exact manner, with a C reference model provided by Intel.[9] - ^ a b c d This instruction cannot be EVEX-encoded. Instead, AVX512F provides different opcodes -
EVEX.66.0F38 4C/4D /r
- for its newVRCP14*
reciprocal approximation instructions.The main difference between the AVX-512
VRCP14*
instructions and the older SSE/AVX(V)RCP*
instructions is that the AVX-512VRRCP14*
instructions have their operation defined in a bit-exact manner, with a C reference model provided by Intel.[9] - ^ a b c d e f g h The EVEX-encoded versions of the
VANDPS
,VANDPD
,VANDNPS
,VANDNPD
,VORPS
,VORPD
,VXORPS
,VXORPD
instructions are not introduced as part of the AVX512F subset, but instead the AVX512DQ subset. - ^
XORPS
/VXORPS
with both source operands being the same register is commonly used as a register-zeroing idiom, and is recognized by most x86 CPUs as an instruction that does not depend on its source arguments.
Under AVX or AVX-512, it is recommended to use a 128-bit form ofVXORPS
for this purpose - this will, on some CPUs, result in fewer micro-ops than wider forms while still achieving register-zeroing of the whole 256 or 512 bit vector-register. - ^ a b c d For EVEX-encoded variants of conversions between FP formats of different widths, the opmask lane width is determined by the result format: 64-bit for
VCVTPS2PD
andVCVTSS2SD
and 32-bit forVCVTPD2PS
andVCVTSS2SD
. - ^ Widening FP→FP conversions (
CVTPS2PD
,CVTSS2SD
,VCVTPH2PD
,VCVTSH2SD
) support the SAE modifier. Narrowing conversions (CVTPD2PS
,CVTSD2SS
) support the RC modifier. - ^ a b For the floating-point minimum-value and maximum-value instructions
(V)MIN*
and(V)MAX*
, if the two input operands are both zero or at least one of the input operands is NaN, then the second input operand is returned. This matches the behavior of common C programming-language expressions such as((op1)>(op2)?(op1):(op2))
for maximum-value and((op1)<(op2)?(op1):(op2))
for minimum-value. - ^ For the SIMD floating-point compares, the imm8 argument has the following format:
Bits Usage 1:0 Basic comparison predicate 2 Invert comparison result 3 Invert comparison result if unordered (VEX/EVEX only) 4 Invert signalling behavior (VEX/EVEX only) The basic comparison predicates are:
Value Meaning 00b Equal (non-signalling) 01b Less-than (signalling) 10b Less-than-or-equal (signalling) 11b Unordered (non-signalling) A signalling compare will cause an exception if any of the inputs are QNaN.
Integer SSE2/4 instructions with 66h prefix, and AVX/AVX-512 extended variants thereof
[edit]These instructions do not have any MMX forms, and do not support any encodings without a prefix. Most of these instructions have extended variants available in VEX-encoded and EVEX-encoded forms:
- The VEX-encoded forms are available under AVX/AVX2. Under AVX, they are available only with a vector length of 128 bits (VEX.L=0 enocding) - under AVX2, they are (with some exceptions noted with "L=0") also made available with a vector length of 256 bits.
- The EVEX-encoded forms are available under AVX-512 - the specific AVX-512 subset needed for each instruction is listed along with the instruction.
Description | Instruction mnemonics | Basic opcode | SSE (66h prefix) | AVX (VEX.66 prefix) |
AVX-512 (EVEX.66 prefix) | ||||
---|---|---|---|---|---|---|---|---|---|
supported | subset | lane | bcst | ||||||
Added with SSE2 | |||||||||
Unpack and interleave low-order 64-bit integers | (V)PUNPCKLQDQ xmm,xmm/m128 [a] |
0F 6C /r |
Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Unpack and interleave high-order 64-bit integers | (V)PUNPCKHQDQ xmm,xmm/m128 [a] |
0F 6D /r |
Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Right-shift 128-bit unsigned integer by specified number of bytes | (V)PSRLDQ xmm,imm8 [a] |
0F 73 /3 ib |
Yes | Yes | Yes | BW | No | No | |
Left-shift 128-bit integer by specified number of bytes | (V)PSLLDQ xmm,imm8 [a] |
0F 73 /7 ib |
Yes | Yes | Yes | BW | No | No | |
Move 64-bit scalar value from xmm register to xmm register or memory | (V)MOVQ xmm/m64,xmm |
0F D6 /r |
Yes | Yes (L=0) | Yes (L=0,W=1) |
F | No | No | |
Added with SSE4.1 | |||||||||
Variable blend packed bytes. For each byte lane of the result, pick the value from either the first or the second argument depending on the top bit of the corresponding byte lane of |
PBLENDVB xmm,xmm/m128 PBLENDVB xmm,xmm/m128,XMM0 [b] |
0F38 10 /r |
Yes | No[c] | No[d] | — | — | — | |
Sign-extend packed integers into wider packed integers | 8-bit → 16-bit | (V)PMOVSXBW xmm,xmm/m64 |
0F38 20 /r |
Yes | Yes | Yes | BW | 16 | No |
8-bit → 32-bit | (V)PMOVSXBD xmm,xmm/m32 |
0F38 21 /r |
Yes | Yes | Yes | F | 32 | No | |
8-bit → 64-bit | (V)PMOVSXBQ xmm,xmm/m16 |
0F38 22 /r |
Yes | Yes | Yes | F | 64 | No | |
16-bit → 32-bit | (V)PMOVSXWD xmm,xmm/m64 |
0F38 23 /r |
Yes | Yes | Yes | F | 32 | No | |
16-bit → 64-bit | (V)PMOVSXWQ xmm,xmm/m32 |
0F38 24 /r |
Yes | Yes | Yes | F | 64 | No | |
32-bit → 64-bit | (V)PMOVSXDQ xmm,xmm/m64 |
0F38 25 /r |
Yes | Yes | Yes (W=0) | F | 64 | No | |
Multiply packed 32-bit signed integers, store full 64-bit result. The input integers are taken from the low 32 bits of each 64-bit vector lane. |
(V)PMULDQ xmm,xmm/m128 |
0F38 28 /r |
Yes | Yes | Yes (W=1) | F | 64 | 64 | |
Compare packed 64-bit integers for equality | (V)PCMPEQQ xmm,xmm/m128 |
0F38 29 /r |
Yes | Yes | Yes (W=1)[e] | F | 64 | 64 | |
Aligned non-temporal vector load from memory.[f] | (V)MOVNTDQA xmm,m128 |
0F38 2A /r |
Yes | Yes | Yes (W=0) | F | No | No | |
Pack 32-bit unsigned integers to 16-bit, with saturation | (V)PACKUSDW xmm, xmm/m128 [a] |
0F38 2B /r |
Yes | Yes | Yes (W=0) | BW | 16 | 32 | |
Zero-extend packed integers into wider packed integers | 8-bit → 16-bit | (V)PMOVZXBW xmm,xmm/m64 |
0F38 30 /r |
Yes | Yes | Yes | BW | 16 | No |
8-bit → 32-bit | (V)PMOVZXBD xmm,xmm/m32 |
0F38 31 /r |
Yes | Yes | Yes | F | 32 | No | |
8-bit → 64-bit | (V)PMOVZXBQ xmm,xmm/m16 |
0F38 32 /r |
Yes | Yes | Yes | F | 64 | No | |
16-bit → 32-bit | (V)PMOVZXWD xmm,xmm/m64 |
0F38 33 /r |
Yes | Yes | Yes | F | 32 | No | |
16-bit → 64-bit | (V)PMOVZXWQ xmm,xmm/m32 |
0F38 34 /r |
Yes | Yes | Yes | F | 64 | No | |
32-bit → 64-bit | (V)PMOVZXDQ xmm,xmm/m64 |
0F38 35 /r |
Yes | Yes | Yes (W=0) | F | 64 | No | |
Packed minimum-value of signed integers | 8-bit | (V)PMINSB xmm,xmm/m128 |
0F38 38 /r |
Yes | Yes | Yes | BW | 8 | No |
32-bit | (V)PMINSD xmm,xmm/m128
|
0F38 39 /r
|
PMINSD
|
VPMINSD
|
VPMINSD (W0) |
F | 32 | 32 | |
64-bit | VPMINSQ xmm,xmm/m128 (AVX-512)
|
VPMINSQ (W1) |
F | 64 | 64 | ||||
Packed minimum-value of unsigned integers | 16-bit | (V)PMINUW xmm,xmm/m128 |
0F38 3A /r |
Yes | Yes | Yes | BW | 16 | No |
32-bit | (V)PMINUD xmm,xmm/m128 |
0F38 3B /r
|
PMINUD
|
VPMINUD
|
VPMINUD (W0) |
F | 32 | 32 | |
64-bit | VPMINUQ xmm,xmm/m128 (AVX-512)
|
VPMINUQ (W1) |
F | 64 | 64 | ||||
Packed maximum-value of signed integers | 8-bit | (V)PMAXSB xmm,xmm/m128 |
0F38 3C /r |
Yes | Yes | Yes | BW | 8 | No |
32-bit | (V)PMAXSD xmm,xmm/m128
|
0F38 3D /r
|
PMAXSD
|
VPMAXSD
|
VPMAXSD (W0) |
F | 32 | 32 | |
64-bit | VPMAXSQ xmm,xmm/m128 (AVX-512)
|
VPMAXSQ (W1) |
F | 64 | 64 | ||||
Packed maximum-value of unsigned integers | 16-bit | (V)PMAXUW xmm,xmm/m128 |
0F38 3E /r |
Yes | Yes | Yes | BW | 16 | No |
32-bit | (V)PMAXUD xmm,xmm/m128 |
0F38 3F /r
|
PMAXUD
|
VPMAXUD
|
VPMAXUD (W0) |
F | 32 | 32 | |
64-bit | VPMAXUQ xmm,xmm/m128 (AVX-512)
|
VPMAXUQ (W1) |
F | 64 | 64 | ||||
Multiply packed 32/64-bit integers, store low half of results | (V)PMULLD mm,mm/m64 PMULLQ xmm,xmm/m128 (AVX-512)
|
0F38 40 /r
|
PMULLD
|
VPMULLD
|
VPMULLD (W0) |
F | 32 | 32 | |
VPMULLQ (W1) |
DQ | 64 | 64 | ||||||
Packed Horizontal Word Minimum Find the smallest 16-bit integer in a packed vector of 16-bit unsigned integers, then return the integer and its index in the bottom two 16-bit lanes of the result vector. |
(V)PHMINPOSUW xmm,xmm/m128 |
0F38 41 /r |
Yes | Yes (L=0) | No | — | — | — | |
Blend Packed Words. For each 16-bit lane of the result, pick a 16-bit value from either the first or the second source argument depending on the corresponding bit of the imm8. |
(V)PBLENDW xmm,xmm/m128,imm8 [a] |
0F3A 0E /r ib |
Yes | Yes[g] | No[h] | — | — | — | |
Extract integer from indexed lane of vector register, and store to GPR or memory. Zero-extended if stored to GPR. |
8-bit | (V)PEXTRB r32/m8,xmm,imm8 [i] |
0F3A 14 /r ib |
Yes | Yes (L=0) | Yes (L=0) | BW | No | No |
16-bit | (V)PEXTRW r32/m16,xmm,imm8 [i] |
0F3A 15 /r ib |
Yes | Yes (L=0) | Yes (L=0) | BW | No | No | |
32-bit | (V)PEXTRD r/m32,xmm,imm8
|
0F3A 16 /r ib
|
Yes | Yes (L=0,W=0)[j] |
Yes (L=0,W=0) |
DQ | No | No | |
64-bit (x86-64) |
(V)PEXTRQ r/m64,xmm,imm8
|
Yes (REX.W) |
Yes (L=0,W=1) |
Yes (L=0,W=1) |
DQ | No | No | ||
Insert integer from general-purpose register into indexed lane of vector register | 8-bit | (V)PINSRB xmm,r32/m8,imm8 [k] |
0F3A 20 /r ib |
Yes | Yes (L=0) | Yes (L=0) | BW | No | No |
32-bit | (V)PINSRD xmm,r32/m32,imm8
|
0F3A 22 /r ib
|
Yes | Yes (L=0,W=0)[j] |
Yes (L=0,W=0) |
DQ | No | No | |
64-bit (x86-64) |
(V)PINSRQ xmm,r64/m64,imm8
|
Yes (REX.W) |
Yes (L=0,W=1) |
Yes (L=0,W=1) |
DQ | No | No | ||
Compute Multiple Packed Sums of Absolute Difference. The 128-bit form of this instruction computes 8 sums of absolute differences from sequentially selected groups of four bytes in the first source argument and a selected group of four contiguous bytes in the second source operand, and writes the sums to sequential 16-bit lanes of destination register. If the two source arguments for i = 0 to 7 do temp[i] := 0 for j = 0 to 3 do a := src1[ i+(imm8[2]*4)+j ] b := src2[ (imm8[1:0]*4)+j ] temp[i] := temp[i] + abs(a-b) done done dst := temp For wider forms of this instruction under AVX2 and AVX10.2, the operation is split into 128-bit lanes where each lane internally performs the same operation as the 128-bit variant of the instruction - except that odd-numbered lanes use bits 5:3 rather than bits 2:0 of the imm8. |
(V)MPSADBW xmm,xmm/m128,imm8 |
0F3A 42 /r ib |
Yes | Yes | Yes (W=0) | 10.2[l] | 16 | No | |
Added with SSE 4.2 | |||||||||
Compare packed 64-bit signed integers for greater-than | (V)PCMPGTQ xmm, xmm/m128 |
0F38 37 /r |
Yes | Yes | Yes (W=1)[e] | F | 64 | 64 | |
Packed Compare Explicit Length Strings, Return Mask | (V)PCMPESTRM xmm,xmm/m128,imm8 |
0F3A 60 /r ib |
Yes[m] | Yes (L=0) | No | — | — | — | |
Packed Compare Explicit Length Strings, Return Index | (V)PCMPESTRI xmm,xmm/m128,imm8 |
0F3A 61 /r ib |
Yes[m] | Yes (L=0) | No | — | — | — | |
Packed Compare Implicit Length Strings, Return Mask | (V)PCMPISTRM xmm,xmm/m128,imm8 |
0F3A 62 /r ib |
Yes[m] | Yes (L=0) | No | — | — | — | |
Packed Compare Implicit Length Strings, Return Index | (V)PCMPISTRI xmm,xmm/m128,imm8 |
0F3A 63 /r ib |
Yes[m] | Yes (L=0) | No | — | — | — |
- ^ a b c d e f For the
(V)PUNPCK*
,(V)PACKUSDW
,(V)PBLENDW
,(V)PSLLDQ
and(V)PSLRDQ
instructions, encodings with a vector-length wider than 128 bits are available under AVX2 and/or AVX-512, but the operation of such encodings is split into 128-bit lanes where each 128-bit lane internally performs the same operation as the 128-bit variant of the instruction. - ^ Assemblers may accept
PBLENDVB
with or withoutXMM0
as a third argument. - ^ The
PBLENDVB
instruction with opcode66 0F38 10 /r
is not VEX-encodable. AVX does provide aVPBLENDVB
instruction that is similar toPBLENDVB
, however, it uses a different opcode and operand encoding -VEX.66.0F3A.W0 4C /r /is4
. - ^ Opcode not EVEX-encodable. Under AVX-512, variable blend of packed bytes may be done with the
VPBLENDMB
instruction (opcodeEVEX.66.0F38.W0 66 /r
). - ^ a b The EVEX-encoded variants of the
VPCMPEQ*
andVPCMPGT*
instructions write their results to AVX-512 opmask registers. This differs from the older non-EVEX variants, which write comparison results as vectors of all-0s/all-1s values to the regular mm/xmm/ymm vector registers. - ^ The load performed by
(V)MOVNTDQA
is weakly-ordered. It may be reordered with respect to other loads, stores and evenLOCK
s - to impose ordering with respect to other loads/stores,MFENCE
or serialization is needed.If
(V)MOVNTDQA
is used with uncached memory, it may fetch a cache-line-sized block of data around the data actually requested - subsequent(V)MOVNTDQA
instructions may return data from blocks fetched in this manner as long as they are not separated by anMFENCE
or serialization. - ^ For AVX, the
VBLENDPS
andVPBLENDD
instructions can be used to perform a blend with 32-bit lanes, allowing one imm8 mask to span a full 256-bit vector without repetition. - ^ Opcode not EVEX-encodable. Under AVX-512, variable blend of packed words may be done with the
VPBLENDMW
instruction (opcodeEVEX.66.0F38.W1 66 /r
). - ^ a b For
(V)PEXTRB
and(V)PEXTRW
, if the destination argument is a register, then the extracted 8/16-bit value is zero-extended to 32/64 bits. - ^ a b For the
VPEXTRD
andVPINSRD
instructions in non-64-bit mode, the instructions are documented as being permitted to be encoded with VEX.W=1 on Intel[10] but not AMD[11] CPUs (although exceptions to this do exist, e.g. Bulldozer permits such encodings[12] while Sandy Bridge does not[13])
In 64-bit mode, these instructions require VEX.W=0 on both Intel and AMD processors — encodings with VEX.W=1 are interpreted asVPEXTRQ
/VPINSRQ
. - ^ In the case of a register source argument to
(V)PINSRB
, the argument is considered to be a 32-bit register of which the 8 bottom bits are used, not an 8-bit register proper. This means that it is not possible to specify AH/BH/CH/DH as a source argument to(V)PINSRB
. - ^ EVEX-encoded variants of the
VMPSADBW
instruction are only available if AVX10.2 is supported. - ^ a b c d The SSE4.2 packed string compare
PCMP*STR*
instructions allow their 16-byte memory operands to be misaligned even when using legacy SSE encoding.
Other SSE/2/3/4 SIMD instructions, and AVX/AVX-512 extended variants thereof
[edit]SSE SIMD instructions that do not fit into any of the preceding groups. Many of these instructions have AVX/AVX-512 extended forms - unless otherwise indicated (L=0 or footnotes) these extended forms support 128/256-bit operation under AVX and 128/256/512-bit operation under AVX-512.
Description | Instruction mnemonics | Basic opcode | SSE | AVX (VEX prefix) |
AVX-512 (EVEX prefix) | |||||
---|---|---|---|---|---|---|---|---|---|---|
supported | subset | lane | bcst | rc/sae | ||||||
Added with SSE | ||||||||||
Load MXCSR (Media eXtension Control and Status Register) from memory | (V)LDMXCSR m32 |
NP 0F AE /2 |
Yes | Yes (L=0) |
No | — | — | — | — | |
Store MXCSR to memory | (V)STMXCSR m32 |
NP 0F AE /3 |
Yes | Yes (L=0) |
No | — | — | — | — | |
Added with SSE2 | ||||||||||
Move a 64-bit data item from MMX register to bottom half of XMM register. Top half is zeroed out. | MOVQ2DQ xmm,mm |
F3 0F D6 /r |
Yes | No | No | — | — | — | — | |
Move a 64-bit data item from bottom half of XMM register to MMX register. | MOVDQ2Q mm,xmm |
F2 0F D6 /r |
Yes | No | No | — | — | — | — | |
Load a 64-bit integer from memory or XMM register to bottom 64 bits of XMM register, with zero-fill | (V)MOVQ xmm,xmm/m64 |
F3 0F 7E /r |
Yes | Yes (L=0) | Yes (L=0,W=1) | F | No | No | No | |
Vector load from unaligned memory or vector register | (V)MOVDQU xmm,xmm/m128 |
F3 0F 6F /r |
Yes | Yes | VMOVDQU64 (W1) |
F | 64 | No | No | |
VMOVDQU32 (W0) |
F | 32 | No | No | ||||||
F2 0F 6F /r |
No | No | VMOVDQU16 (W1) |
BW | 16 | No | No | |||
VMOVDQU8 (W0) |
BW | 8 | No | No | ||||||
Vector store to unaligned memory or vector register | (V)MOVDQU xmm/m128,xmm |
F3 0F 7F /r |
Yes | Yes | VMOVDQU64 (W1) |
F | 64 | No | No | |
VMOVDQU32 (W0) |
F | 32 | No | No | ||||||
F2 0F 7F /r |
No | No | VMOVDQU16 (W1) |
BW | 16 | No | No | |||
VMOVDQU8 (W0) |
BW | 8 | No | No | ||||||
Shuffle the four top 16-bit lanes of source vector, then place result in top half of destination vector | (V)PSHUFHW xmm,xmm/m128,imm8 [a] |
F3 0F 70 /r ib |
Yes | Yes[b] | Yes | BW | 16 | No | No | |
Shuffle the four bottom 16-bit lanes of source vector, then place result in bottom half of destination vector | (V)PSHUFLW xmm,xmm/m128,imm8 [a] |
F2 0F 70 /r ib |
Yes | Yes[b] | Yes | BW | 16 | No | No | |
Convert packed signed 32-bit integers to FP32 | (V)CVTDQ2PS xmm,xmm/m128 |
NP 0F 5B /r |
Yes | Yes | Yes (W=0) | F | 32 | 32 | RC | |
Convert packed FP32 values to packed signed 32-bit integers | (V)CVTPS2DQ xmm,xmm/m128 |
66 0F 5B /r |
Yes | Yes | Yes (W=0) | F | 32 | 32 | RC | |
Convert packed FP32 values to packed signed 32-bit integers, with round-to-zero | (V)CVTTPS2DQ xmm,xmm/m128 |
F3 0F 5B /r |
Yes | Yes | Yes (W=0) | F | 32 | 32 | SAE | |
Convert packed FP64 values to packed signed 32-bit integers, with round-to-zero | (V)CVTTPD2DQ xmm,xmm/m64 |
66 0F E6 /r |
Yes | Yes | Yes (W=1) | F | 32 | 64 | SAE | |
Convert packed signed 32-bit integers to FP64 | (V)CVTDQ2PD xmm,xmm/m64 |
F3 0F E6 /r |
Yes | Yes | Yes (W=0) | F | 64 | 32 | RC[c] | |
Convert packed FP64 values to packed signed 32-bit integers | (V)CVTPD2DQ xmm,xmm/m128 |
F2 0F E6 /r |
Yes | Yes | Yes (W=1) | F | 32 | 64 | RC | |
Added with SSE3 | ||||||||||
Duplicate floating-point values from even-numbered lanes to next odd-numbered lanes up | 32-bit | (V)MOVSLDUP xmm,xmm/m128 |
F3 0F 12 /r |
Yes | Yes | Yes (W=0) | F | 32 | No | No |
64-bit | (V)MOVDDUP xmm/xmm/m128 |
F2 0F 12 /r |
Yes | Yes | Yes (W=1) | F | 64 | No | No | |
Duplicate FP32 values from odd-numbered lanes to next even-numbered lanes down | (V)MOVSHDUP xmm,xmm/m128 |
F3 0F 16 /r |
Yes | Yes | Yes (W=0) | F | 32 | No | No | |
Packed pairwise horizontal addition of floating-point values | 32-bit | (V)HADDPS xmm,xmm/m128 [a] |
F2 0F 7C /r |
Yes | Yes | No | — | — | — | — |
64-bit | (V)HADDPD xmm,xmm/m128 [a] |
66 0F 7C /r |
Yes | Yes | No | — | — | — | — | |
Packed pairwise horizontal subtraction of floating-point values | 32-bit | (V)HSUBPS xmm,xmm/m128 [a] |
F2 0F 7D /r |
Yes | Yes | No | — | — | — | — |
64-bit | (V)HSUBPD xmm,xmm/m128 [a] |
66 0F 7D /r |
Yes | Yes | No | — | — | — | — | |
Packed floating-point add/subtract in alternating lanes. Even-numbered lanes (counting from 0) do subtract, odd-numbered lanes do add. | 32-bit | (V)ADDSUBPS xmm,xmm/m128 |
F2 0F D0 /r |
Yes | Yes | No | — | — | — | — |
64-bit | (V)ADDSUBPD xmm,xmm/m128 |
66 0F D0 /r |
Yes | Yes | No | — | — | — | — | |
Vector load from unaligned memory with looser semantics than (V)MOVDQU .Unlike |
(V)LDDQU xmm,m128 |
F2 0F F0 /r |
Yes | Yes | No | — | — | — | — | |
Added with SSE4.1 | ||||||||||
Vector logical test. Sets ZF=1 if bitwise-AND between first operand and second operand results in all-0s, ZF=0 otherwise. Sets CF=1 if bitwise-AND between second operand and bitwise-NOT of first operand results in all-0s, CF=0 otherwise |
(V)PTEST xmm,xmm/m128 |
66 0F38 17 /r |
Yes | Yes | No[d] | — | — | — | — | |
Variable blend packed floating-point values. For each lane of the result, pick the value from either the first or the second argument depending on the top bit of the corresponding lane of |
32-bit | BLENDVPS xmm,xmm/m128 BLENDVPS xmm,xmm/m128,XMM0 [e] |
66 0F38 14 /r |
Yes | No[f] | No | — | — | — | — |
64-bit | BLENDVPD xmm,xmm/m128 BLENDVPD xmm,xmm/m128,XMM0 [e] |
66 0F38 15 /r |
Yes | No[f] | No | — | — | — | — | |
Rounding of packed floating-point values to integer. Rounding mode specified by imm8 argument. |
32-bit | (V)ROUNDPS xmm,xmm/m128,imm8 |
66 0F3A 08 /r ib |
Yes | Yes | No[g] | — | — | — | — |
64-bit | (V)ROUNDPD xmm,xmm/m128,imm8 |
66 0F3A 09 /r ib |
Yes | Yes | No[g] | — | — | — | — | |
Rounding of scalar floating-point value to integer. | 32-bit | (V)ROUNDSS xmm,xmm/m128,imm8 |
66 0F3A 0A /r ib |
Yes | Yes | No[g] | — | — | — | — |
64-bit | (V)ROUNDSD xmm,xmm/m128,imm8 |
66 0F3A 0B /r ib |
Yes | Yes | No[g] | — | — | — | — | |
Blend packed floating-point values. For each lane of the result, pick the value from either the first or the second argument depending on the corresponding imm8 bit. | 32-bit | (V)BLENDPS xmm,xmm/m128,imm8 |
66 0F3A 0C /r ib |
Yes | Yes | No | — | — | — | — |
64-bit | (V)BLENDPD xmm,xmm/m128,imm8 |
66 0F3A 0D /r ib |
Yes | Yes | No | — | — | — | — | |
Extract 32-bit lane of XMM register to general-purpose register or memory location. Bits[1:0] of imm8 is used to select lane. |
(V)EXTRACTPS r/m32,xmm,imm8 |
66 0F3A 17 /r ib |
Yes | Yes (L=0) | Yes (L=0) | F | No | No | No | |
Obtain 32-bit value from source XMM register or memory, and insert into the specified lane of destination XMM register. If the source argument is an XMM register, then bits[7:6] of the imm8 is used to select which 32-bit lane to select source from, otherwise the specified 32-bit memory value is used. This 32-bit value is then inserted into the destination register lane specified by bits[5:4] of the imm8. After insertion, each 32-bit lane of the destination register may optionally be zeroed out - bits[3:0] of the imm8 provides a bitmap of which lanes to zero out. |
(V)INSERTPS xmm,xmm/m32,imm8 |
66 0F3A 21 /r ib |
Yes | Yes (L=0) | Yes (L=0,W=0) | F | No | No | No | |
4-component dot-product of 32-bit floating-point values. Bits [7:4] of the imm8 specify which lanes should participate in the dot-product, bits[3:0] specify which lanes in the result should receive the dot-product (remaining lanes are filled with zeros) |
(V)DPPS xmm,xmm/m128,imm8 [a] |
66 0F3A 40 /r ib |
Yes | Yes | No | — | — | — | — | |
2-component dot-product of 64-bit floating-point values. Bits [5:4] of the imm8 specify which lanes should participate in the dot-product, bits[1:0] specify which lanes in the result should receive the dot-product (remaining lanes are filled with zeros) |
(V)DPPD xmm,xmm/m128,imm8 [a] |
66 0F3A 41 /r ib |
Yes | Yes | No | — | — | — | — | |
Added with SSE4a (AMD only) | ||||||||||
64-bit bitfield insert, using the low 64 bits of XMM registers. First argument is an XMM register to insert bitfield into, second argument is a source register containing the bitfield to insert (starting from bit 0). For the 4-argument version, the first imm8 specifies bitfield length and the second imm8 specifies bit-offset to insert bitfield at. For the 2-argument version, the length and offset are instead taken from bits [69:64] and [77:72] of the second argument, respectively. |
INSERTQ xmm,xmm,imm8,imm8 |
F2 0F 78 /r ib ib |
Yes | No | No[h] | — | — | — | — | |
INSERTQ xmm,xmm |
F2 0F 79 /r |
Yes | No | No[h] | — | — | — | — | ||
64-bit bitfield extract, from the lower 64 bits of an XMM register. The first argument serves as both source that bitfield is extracted from and destination that bitfield is written to. For the 3-argument version, the first imm8 specifies bitfield length and the second imm8 specifies bitfield bit-offset. For the 2-argument version, the second argument is an XMM register that contains bitfield length at bits[5:0] and bit-offset at bits[13:8]. |
EXTRQ xmm,imm8,imm8 |
66 0F 78 /0 ib ib |
Yes | No | No[h] | — | — | — | — | |
EXTRQ xmm,xmm |
66 0F 79 /r |
Yes | No | No[h] | — | — | — | — |
- ^ a b c d e f g h For the
VPSHUFLW
,VPSHUFHW
,VHADDP*
,VHSUBP*
,VDPPS
andVDPPD
instructions, encodings with a vector-length wider than 128 bits are available under AVX2 and/or AVX-512, but the operation of such encodings is split into 128-bit lanes where each 128-bit lane internally performs the same operation as the 128-bit variant of the instruction. - ^ a b Under AVX, the
VPSHUFHW
andVPSHUFLW
instructions are only available in 128-bit forms - the 256-bit forms of these instructions require AVX2. - ^ For the EVEX-encoded form of
VCVTDQ2PD
, EVEX embedded rounding controls are permitted but have no effect. - ^ Opcode not EVEX-encodable. Performing a vector logical test under AVX-512 requires a sequence of at least 2 instructions, e.g.
VPTESTMD
followed byKORTESTW
. - ^ a b Assemblers may accept the
BLENDVPS
/BLENDVPD
instructions with or withoutXMM0
as a third argument. - ^ a b While AVX does provide
VBLENDVPS/VPD
instruction that are similar in function toBLENDVPS/VPD
, they uses a different opcode and operand encoding -VEX.66.0F3A.W0 4A/4B /r /is4
. - ^ a b c d Opcode not available under AVX-512. Instead, AVX512F provides different opcodes -
EVEX.66.0F3A (08..0B) /r ib
- for its newVRNDSCALE*
rounding instructions. - ^ a b c d Under AVX-512, EVEX-encoding the
INSERTQ
/EXTRQ
opcodes result in AVX-512 instructions completely unrelated to SSE4a, namelyVCVT(T)P(S|D)2UQQ
andVCVT(T)S(S|D)2USI
.
AVX were first supported by Intel with Sandy Bridge and by AMD with Bulldozer.
Vector operations on 256 bit registers.
Instruction | Description |
---|---|
VBROADCASTSS | Copy a 32-bit, 64-bit or 128-bit memory operand to all elements of a XMM or YMM vector register. |
VBROADCASTSD | |
VBROADCASTF128 | |
VINSERTF128 | Replaces either the lower half or the upper half of a 256-bit YMM register with the value of a 128-bit source operand. The other half of the destination is unchanged. |
VEXTRACTF128 | Extracts either the lower half or the upper half of a 256-bit YMM register and copies the value to a 128-bit destination operand. |
VMASKMOVPS | Conditionally reads any number of elements from a SIMD vector memory operand into a destination register, leaving the remaining vector elements unread and setting the corresponding elements in the destination register to zero. Alternatively, conditionally writes any number of elements from a SIMD vector register operand to a vector memory operand, leaving the remaining elements of the memory operand unchanged. On the AMD Jaguar processor architecture, this instruction with a memory source operand takes more than 300 clock cycles when the mask is zero, in which case the instruction should do nothing. This appears to be a design flaw.[14] |
VMASKMOVPD | |
VPERMILPS | Permute In-Lane. Shuffle the 32-bit or 64-bit vector elements of one input operand. These are in-lane 256-bit instructions, meaning that they operate on all 256 bits with two separate 128-bit shuffles, so they can not shuffle across the 128-bit lanes.[15] |
VPERMILPD | |
VPERM2F128 | Shuffle the four 128-bit vector elements of two 256-bit source operands into a 256-bit destination operand, with an immediate constant as selector. |
VZEROALL | Set all YMM registers to zero and tag them as unused. Used when switching between 128-bit use and 256-bit use. |
VZEROUPPER | Set the upper half of all YMM registers to zero. Used when switching between 128-bit use and 256-bit use. |
Half-precision floating-point conversion.
Instruction | Meaning |
---|---|
VCVTPH2PS xmmreg,xmmrm64 | Convert four half-precision floating point values in memory or the bottom half of an XMM register to four single-precision floating-point values in an XMM register |
VCVTPH2PS ymmreg,xmmrm128 | Convert eight half-precision floating point values in memory or an XMM register (the bottom half of a YMM register) to eight single-precision floating-point values in a YMM register |
VCVTPS2PH xmmrm64,xmmreg,imm8 | Convert four single-precision floating point values in an XMM register to half-precision floating-point values in memory or the bottom half an XMM register |
VCVTPS2PH xmmrm128,ymmreg,imm8 | Convert eight single-precision floating point values in a YMM register to half-precision floating-point values in memory or an XMM register |
Introduced in Intel's Haswell microarchitecture and AMD's Excavator.
Expansion of most vector integer SSE and AVX instructions to 256 bits
Instruction | Description |
---|---|
VBROADCASTSS | Copy a 32-bit or 64-bit register operand to all elements of a XMM or YMM vector register. These are register versions of the same instructions in AVX1. There is no 128-bit version however, but the same effect can be simply achieved using VINSERTF128. |
VBROADCASTSD | |
VPBROADCASTB | Copy an 8, 16, 32 or 64-bit integer register or memory operand to all elements of a XMM or YMM vector register. |
VPBROADCASTW | |
VPBROADCASTD | |
VPBROADCASTQ | |
VBROADCASTI128 | Copy a 128-bit memory operand to all elements of a YMM vector register. |
VINSERTI128 | Replaces either the lower half or the upper half of a 256-bit YMM register with the value of a 128-bit source operand. The other half of the destination is unchanged. |
VEXTRACTI128 | Extracts either the lower half or the upper half of a 256-bit YMM register and copies the value to a 128-bit destination operand. |
VGATHERDPD | Gathers single or double precision floating point values using either 32 or 64-bit indices and scale. |
VGATHERQPD | |
VGATHERDPS | |
VGATHERQPS | |
VPGATHERDD | Gathers 32 or 64-bit integer values using either 32 or 64-bit indices and scale. |
VPGATHERDQ | |
VPGATHERQD | |
VPGATHERQQ | |
VPMASKMOVD | Conditionally reads any number of elements from a SIMD vector memory operand into a destination register, leaving the remaining vector elements unread and setting the corresponding elements in the destination register to zero. Alternatively, conditionally writes any number of elements from a SIMD vector register operand to a vector memory operand, leaving the remaining elements of the memory operand unchanged. |
VPMASKMOVQ | |
VPERMPS | Shuffle the eight 32-bit vector elements of one 256-bit source operand into a 256-bit destination operand, with a register or memory operand as selector. |
VPERMD | |
VPERMPD | Shuffle the four 64-bit vector elements of one 256-bit source operand into a 256-bit destination operand, with a register or memory operand as selector. |
VPERMQ | |
VPERM2I128 | Shuffle (two of) the four 128-bit vector elements of two 256-bit source operands into a 256-bit destination operand, with an immediate constant as selector. |
VPBLENDD | Doubleword immediate version of the PBLEND instructions from SSE4. |
VPSLLVD | Shift left logical. Allows variable shifts where each element is shifted according to the packed input. |
VPSLLVQ | |
VPSRLVD | Shift right logical. Allows variable shifts where each element is shifted according to the packed input. |
VPSRLVQ | |
VPSRAVD | Shift right arithmetically. Allows variable shifts where each element is shifted according to the packed input. |
FMA3 and FMA4 instructions
[edit]Floating-point fused multiply-add instructions are introduced in x86 as two instruction set extensions, "FMA3" and "FMA4", both of which build on top of AVX to provide a set of scalar/vector instructions using the xmm/ymm/zmm vector registers. FMA3 defines a set of 3-operand fused-multiply-add instructions that take three input operands and writes its result back to the first of them. FMA4 defines a set of 4-operand fused-multiply-add instructions that take four input operands – a destination operand and three source operands.
FMA3 is supported on Intel CPUs starting with Haswell, on AMD CPUs starting with Piledriver, and on Zhaoxin CPUs starting with YongFeng. FMA4 was only supported on AMD Family 15h (Bulldozer) CPUs and has been abandoned from AMD Zen onwards. The FMA3/FMA4 extensions are not considered to be an intrinsic part of AVX or AVX2, although all Intel and AMD (but not Zhaoxin) processors that support AVX2 also support FMA3. FMA3 instructions (in EVEX-encoded form) are, however, AVX-512 foundation instructions.
The FMA3 and FMA4 instruction sets both define a set of 10 fused-multiply-add operations, all available in FP32 and FP64 variants. For each of these variants, FMA3 defines three operand orderings while FMA4 defines two.
FMA3 encoding
FMA3 instructions are encoded with the VEX or EVEX prefixes – on the form VEX.66.0F38 xy /r
or EVEX.66.0F38 xy /r
. The VEX.W/EVEX.W bit selects floating-point format (W=0 means FP32, W=1 means FP64). The opcode byte xy
consists of two nibbles, where the top nibble x
selects operand ordering (9
='132', A
='213', B
='231') and the bottom nibble y
(values 6..F) selects which one of the 10 fused-multiply-add operations to perform. (x
and y
outside the given ranges will result in something that is not an FMA3 instruction.)
At the assembly language level, the operand ordering is specified in the mnemonic of the instruction:
vfmadd132sd xmm1,xmm2,xmm3
will performxmm1 ← (xmm1*xmm3)+xmm2
vfmadd213sd xmm1,xmm2,xmm3
will performxmm1 ← (xmm2*xmm1)+xmm3
vfmadd231sd xmm1,xmm2,xmm3
will performxmm1 ← (xmm2*xmm3)+xmm1
For all FMA3 variants, the first two arguments must be xmm/ymm/zmm vector register arguments, while the last argument may be either a vector register or memory argument. Under AVX-512 and AVX10, the EVEX-encoded variants support EVEX-prefix-encoded broadcast, opmasks and rounding-controls.
The AVX512-FP16 extension, introduced in Sapphire Rapids, adds FP16 variants of the FMA3 instructions – these all take the form EVEX.66.MAP6.W0 xy /r
with the opcode byte working in the same way as for the FP32/FP64 variants. The AVX10.2 extension, published in 2024,[16] similarly adds BF16 variants of the packed (but not scalar) FMA3 instructions – these all take the form EVEX.NP.MAP6.W0 xy /r
with the opcode byte again working similar to the FP32/FP64 variants.
(For the FMA4 instructions, no FP16 or BF16 variants are defined.)
FMA4 encoding
FMA4 instructions are encoded with the VEX prefix, on the form VEX.66.0F3A xx /r ib
(no EVEX encodings are defined). The opcode byte xx
uses its bottom bit to select floating-point format (0=FP32, 1=FP64) and the remaining bits to select one of the 10 fused-multiply-add operations to perform.
For FMA4, operand ordering is controlled by the VEX.W bit. If VEX.W=0, then the third operand is the r/m operand specified by the instruction's ModR/M byte and the fourth operand is a register operand, specified by bits 7:4 of the ib (8-bit immediate) part of the instruction. If VEX.W=1, then these two operands are swapped. For example:
vfmaddsd xmm1,xmm2,[mem],xmm3
will performxmm1 ← (xmm2*[mem])+xmm3
and require a W=0 encoding.vfmaddsd xmm1,xmm2,xmm3,[mem]
will performxmm1 ← (xmm2*xmm3)+[mem]
and require a W=1 encoding.vfmaddsd xmm1,xmm2,xmm3,xmm4
will performxmm1 ← (xmm2*xmm3)+xmm4
and can be encoded with either W=0 or W=1.
Opcode table
The 10 fused-multiply-add operations and the 122 instruction variants they give rise to are given by the following table – with FMA4 instructions highlighted with * and yellow cell coloring, and FMA3 instructions not highlighted:
Basic operation | Opcode byte | FP32 instructions | FP64 instructions | FP16 instructions (AVX512-FP16) |
BF16 instructions (AVX10.2) |
---|---|---|---|---|---|
Packed alternating multiply-add/subtract
|
96 |
VFMADDSUB132PS |
VFMADDSUB132PD |
VFMADDSUB132PH |
— |
A6 |
VFMADDSUB213PS |
VFMADDSUB213PD |
VFMADDSUB213PH |
— | |
B6 |
VFMADDSUB231PS |
VFMADDSUB231PD |
VFMADDSUB231PH |
— | |
5C/5D * |
VFMADDSUBPS * |
VFMADDSUBPD * |
— | — | |
Packed alternating multiply-subtract/add
|
97 |
VFMSUBADD132PS |
VFMSUBADD132PD |
VFMSUBADD132PH |
— |
A7 |
VFMSUBADD213PS |
VFMSUBADD213PD |
VFMSUBADD213PH |
— | |
B7 |
VFMSUBADD231PS |
VFMSUBADD231PD |
VFMSUBADD231PH |
— | |
5E/5F * |
VFMSUBADDPS * |
VFMSUBADDPD * |
— | — | |
Packed multiply-add (A*B)+C |
98 |
VFMADD132PS |
VFMADD132PD |
VFMADD132PH |
VFMADD132BF16
|
A8 |
VFMADD213PS |
VFMADD213PD |
VFMADD213PH |
VFMADD213BF16
| |
B8 |
VFMADD231PS |
VFMADD231PD |
VFMADD231PH |
VFMADD231BF16
| |
68/69 * |
VFMADDPS * |
VFMADDPD * |
— | — | |
Scalar multiply-add (A*B)+C |
99 |
VFMADD132SS |
VFMADD132SD |
VFMADD132SH |
— |
A9 |
VFMADD213SS |
VFMADD213SD |
VFMADD213SH |
— | |
B9 |
VFMADD231SS |
VFMADD231SD |
VFMADD231SH |
— | |
6A/6B * |
VFMADDSS * |
VFMADDSD * |
— | — | |
Packed multiply-subtract (A*B)-C |
9A |
VFMSUB132PS |
VFMSUB132PD |
VFMSUB132PH |
VFMSUB132BF16
|
AA |
VFMSUB213PS |
VFMSUB213PD |
VFMSUB213PH |
VFMSUB213BF16
| |
BA |
VFMSUB231PS |
VFMSUB231PD |
VFMSUB231PH |
VFMSUB231BF16
| |
6C/6D * |
VFMSUBPS * |
VFMSUBPD * |
— | — | |
Scalar multiply-subtract (A*B)-C |
9B |
VFMSUB132SS |
VFMSUB132SD |
VFMSUB132SH |
— |
AB |
VFMSUB213SS |
VFMSUB213SD |
VFMSUB213SH |
— | |
BB |
VFMSUB231SS |
VFMSUB231SD |
VFMSUB231SH |
— | |
6E/6F * |
VFMSUBSS * |
VFMSUBSD * |
— | — | |
Packed negative-multiply-add (-A*B)+C |
9C |
VFNMADD132PS |
VFNMADD132PD |
VFNMADD132PH |
VFNMADD132BF16
|
AC |
VFNMADD213PS |
VFNMADD213PD |
VFNMADD213PH |
VFNMADD213BF16
| |
BC |
VFNMADD231PS |
VFNMADD231PD |
VFNMADD231PH |
VFNMADD231BF16
| |
78/79 * |
VFMADDPS * |
VFMADDPD * |
— | — | |
Scalar negative-multiply-add (-A*B)+C |
9D |
VFMADD132SS |
VFMADD132SD |
VFMADD132SH |
— |
AD |
VFMADD213SS |
VFMADD213SD |
VFMADD213SH |
— | |
BD |
VFMADD231SS |
VFMADD231SD |
VFMADD231SH |
— | |
7A/7B * |
VFMADDSS * |
VFMADDSD * |
— | — | |
Packed negative-multiply-subtract (-A*B)-C |
9E |
VFNMSUB132PS |
VFNMSUB132PD |
VFNMSUB132PH |
VFNMSUB132BF16
|
AE |
VFNMSUB213PS |
VFNMSUB213PD |
VFNMSUB213PH |
VFNMSUB213BF16
| |
BE |
VFNMSUB231PS |
VFNMSUB231PD |
VFNMSUB231PH |
VFNMSUB231BF16
| |
7C/7D * |
VFNMSUBPS * |
VFNMSUBPD * |
— | — | |
Scalar negative-multiply-subtract (-A*B)-C |
9F |
VFNMSUB132SS |
VFNMSUB132SD |
VFNMSUB132SH |
— |
AF |
VFNMSUB213SS |
VFNMSUB213SD |
VFNMSUB213SH |
— | |
BF |
VFNMSUB231SS |
VFNMSUB231SD |
VFNMSUB231SH |
— | |
7E/7F * |
VFNMSUBSS * |
VFNMSUBSD * |
— | — |
- ^ Vector register lanes are counted from 0 upwards in a little-endian manner – the lane that contains the first byte of the vector is considered to be even-numbered.
AVX-512
[edit]AVX-512, introduced in 2014, adds 512-bit wide vector registers (extending the 256-bit registers, which become the new registers' lower halves) and doubles their count to 32; the new registers are thus named zmm0 through zmm31. It adds eight mask registers, named k0 through k7, which may be used to restrict operations to specific parts of a vector register. Unlike previous instruction set extensions, AVX-512 is implemented in several groups; only the foundation ("AVX-512F") extension is mandatory.[17] Most of the added instructions may also be used with the 256- and 128-bit registers.
AMX
[edit]Intel AMX adds eight new tile-registers, tmm0
-tmm7
, each holding a matrix, with a maximum capacity of 16 rows of 64 bytes per tile-register. It also adds a TILECFG
register to configure the sizes of the actual matrices held in each of the eight tile-registers, and a set of instructions to perform matrix multiplications on these registers.
AMX subset | Instruction mnemonics | Opcode | Instruction description | Added in |
---|---|---|---|---|
|
LDTILECFG m512 |
VEX.128.NP.0F38.W0 49 /0 |
Load AMX tile configuration data structure from memory as a 64-byte data structure. | Sapphire Rapids |
STTILECFG m512 |
VEX.128.66.0F38 W0 49 /0 |
Store AMX tile configuration data structure to memory. | ||
TILERELEASE |
VEX.128.NP.0F38.W0 49 C0 |
Initialize TILECFG and tile data registers (tmm0 to tmm7 ) to the INIT state (all-zeroes).
| ||
TILEZERO tmm |
VEX.128.F2.0F38.W0 49 /r [a] |
Zero out contents of one tile register. | ||
TILELOADD tmm, sibmem |
VEX.128.F2.0F38.W0 4B /r [b] |
Load a data tile from memory into AMX tile register. | ||
TILELOADDT1 tmm, sibmem |
VEX.128.66.0F38.W0 4B /r [b] |
Load a data tile from memory into AMX tile register, with a hint that data should not be kept in the nearest cache levels. | ||
TILESTORED mem, sibtmm |
VEX.128.F3.0F38.W0 4B /r [b] |
Store a data tile to memory from AMX tile register. | ||
|
TDPBSSD tmm1,tmm2,tmm3 [c] |
VEX.128.F2.0F38.W0 5E /r |
Matrix multiply signed bytes from tmm2 with signed bytes from tmm3, accumulating result in tmm1. | |
TDPBSUD tmm1,tmm2,tmm3 [c] |
VEX.128.F3.0F38.W0 5E /r |
Matrix multiply signed bytes from tmm2 with unsigned bytes from tmm3, accumulating result in tmm1. | ||
TDPBUSD tmm1,tmm2,tmm3 [c] |
VEX.128.66.0F38.W0 5E /r |
Matrix multiply unsigned bytes from tmm2 with signed bytes from tmm3, accumulating result in tmm1. | ||
TDPBUUD tmm1,tmm2,tmm3 [c] |
VEX.128.NP.0F38.W0 5E /r |
Matrix multiply unsigned bytes from tmm2 with unsigned bytes from tmm3, accumulating result in tmm1. | ||
TDPBF16PS tmm1,tmm2,tmm3 [c] |
VEX.128.F3.0F38.W0 5C /r |
Matrix multiply BF16 values from tmm2 with BF16 values from tmm3, accumulating result in tmm1. | ||
TDPFP16PS tmm1,tmm2,tmm3 [c] |
VEX.128.F2.0F38.W0 5C /r |
Matrix multiply FP16 values from tmm2 with FP16 values from tmm3, accumulating result in tmm1. | (Granite Rapids) | |
|
TCMMRLFP16PS tmm1,tmm2,tmm3 [c] |
VEX.128.NP.0F38.W0 6C /r |
Matrix multiply complex numbers from tmm2 with complex numbers from tmm3, accumulating real part of result in tmm1. | (Granite Rapids D) |
TCMMILFP16PS tmm1,tmm2,tmm3 [c] |
VEX.128.66.0F38.W0 6C /r |
Matrix multiply complex numbers from tmm2 with complex numbers from tmm3, accumulating imaginary part of result in tmm1. |
- ^ For
TILEZERO
, the tile-register to clear is specified by bits 5:3 of the instruction's ModR/M byte. Bits 7:6 must be set to 11b, and bits 2:0 must be set to 000b. - ^ a b c For the
TILELOADD
,TILELOADDT1
andTILESTORED
instructions, the memory argument must use a memory addressing mode with the SIB-byte. Under this addressing mode, the base register and displacement are used to specify the starting address for the first row of the tile to load/store from/to memory – the scale and index are used to specify a per-row stride.
These instructions are all interruptible – an interrupt or memory exception taken in the middle of these instructions will cause progress tracking information to be written toTILECFG.start_row
, so that the instruction may continue on a partially-loaded/stored tile after the interruption. - ^ a b c d e f g h For all of the AMX matrix multiply instructions, the three arguments are required to be three different tile registers, or else the instruction will #UD.
See also
[edit]References
[edit]- ^ Intel, Avoiding AVX-SSE Transition Penalties, see section 3.3. Archived on Sep 20, 2024.
- ^ Intel, 2nd Generation Intel Core Processor Family Desktop Specification Update, order no. 324643-037, apr 2016, see erratum BJ49 on page 36. Archived from the original on 6 Jul 2017.
- ^ Intel, Intel 64 and IA-32 Architectures Software Developer’s Manual order no. 325462-086, December 2024, Volume 2B, MOVD/MOVQ instruction entry, page 1289. Archived on 30 Dec 2024.
- ^ AMD, AMD64 Architecture Programmer’s Manual Volumes 1–5, pub.no. 40332, rev 4.08, April 2024, see entries for
MOVD
instruction on pages 2159 and 3040. Archived on 19 Jan 2025. - ^ Intel, Intel 64 and IA-32 Architectures Software Developer’s Manual order no. 325462-086, December 2024, Volume 3B section 10.1.1, page 3368. Archived on 30 Dec 2024.
- ^ AMD, AMD64 Architecture Programmer’s Manual Volumes 1–5, pub.no. 40332, rev 4.08, April 2024, Volume 2, section 7.3.2 on page 650. Archived on 19 Jan 2025.
- ^ GCC Bugzilla, Bug 104688 - gcc and libatomic can use SSE for 128-bit atomic loads on Intel and AMD CPUs with AVX, see comments 34 and 38 for a statement from Zhaoxin on
VMOVDQA
atomicity. Archived on 12 Dec 2024. - ^ Stack Overflow, SSE instructions: which CPUs can do atomic 16B memory operations? Archived on 30 Sep 2024.
- ^ a b Intel, Reference Implementations for Intel Architecture Approximation Instructions VRCP14, VRSQRT14, VRCP28, VRSQRT28, and VEXP2, id #671685, Dec 28, 2015. Archived on Sep 18, 2023.
C code "recip14.c" archived on 18 Sep 2023.
- ^ Intel, Intel 64 and IA-32 Architectures Software Developer’s Manual order no. 325462-086, December 2024, Volume 2B,
VPEXTRD
entry on page 1511 andVPINSRD
entry on page 1530. Archived on 30 Dec 2024. - ^ AMD, AMD64 Architecture Programmer’s Manual Volumes 1–5, pub.no. 40332, rev 4.08, April 2024, see entry for
VPEXTRD
on page 2302 andVPINSRD
on page 2329. Archived on 19 Jan 2025. - ^ AMD, Revision Guide for AMD Family 15h Models 00h-0Fh Processors, pub.no. 38603 rev 3.24, sep 2014, see erratum 592 on page 37. Archived on 22 Jan 2025.
- ^ Intel, 2nd Generation Intel Core Processor Family Desktop Specification Update, order no. 324643-037, apr 2016, see erratum BJ72 on page 43. Archived from the original on 6 Jul 2017.
- ^ "The microarchitecture of Intel, AMD and VIA CPUs: An optimization guide for assembly programmers and compiler makers" (PDF). Retrieved October 17, 2016.
- ^ "Chess programming AVX2". Archived from the original on July 10, 2017. Retrieved October 17, 2016.
- ^ Intel, Advanced Vector Extensions 10.2 Architecture Specification, order no. 361050-001, rev 1.0, July 2024. Archived on 1 Aug 2024.
- ^ "Intel AVX-512 Instructions". Intel. Retrieved 21 June 2022.
External links
[edit]- Intel Intrinsics Guide - searchable reference for Intel MMX/SSE/AVX/AVX512 SIMD intrinsics