Opcode/Instruction | Op/En | 64/32 -bit Mode | CPUID Feature Flag | Description |
---|---|---|---|---|
VEX.128.66.0F38.W0 90 /r VPGATHERDD xmm1, vm32x, xmm2 | RMV | V/V | AVX2 | Using dword indices specified in vm32x, gather dword values from memory conditioned on mask specified by xmm2. Conditionally gathered elements are merged into xmm1. |
VEX.128.66.0F38.W0 91 /r VPGATHERQD xmm1, vm64x, xmm2 | RMV | V/V | AVX2 | Using qword indices specified in vm64x, gather dword values from memory conditioned on mask specified by xmm2. Conditionally gathered elements are merged into xmm1. |
VEX.256.66.0F38.W0 90 /r VPGATHERDD ymm1, vm32y, ymm2 | RMV | V/V | AVX2 | Using dword indices specified in vm32y, gather dword from memory conditioned on mask specified by ymm2. Conditionally gathered elements are merged into ymm1. |
VEX.256.66.0F38.W0 91 /r VPGATHERQD xmm1, vm64y, xmm2 | RMV | V/V | AVX2 | Using qword indices specified in vm64y, gather dword values from memory conditioned on mask specified by xmm2. Conditionally gathered elements are merged into xmm1. |
Op/En | Operand 1 | Operand 2 | Operand 3 | Operand 4 |
---|---|---|---|---|
RMV | ModRM:reg (r,w) | BaseReg (R): VSIB:base, VectorReg(R): VSIB:index | VEX.vvvv (r, w) | N/A |
The instruction conditionally loads up to 4 or 8 dword values from memory addresses specified by the memory operand (the second operand) and using dword indices. The memory operand uses the VSIB form of the SIB byte to specify a general purpose register operand as the common base, a vector register for an array of indices relative to the base and a constant scale factor.
The mask operand (the third operand) specifies the conditional load operation from each memory address and the corresponding update of each data element of the destination operand (the first operand). Conditionality is specified by the most significant bit of each data element of the mask register. If an element’s mask bit is not set, the corresponding element of the destination register is left unchanged. The width of data element in the destination register and mask register are identical. The entire mask register will be set to zero by this instruction unless the instruction causes an exception.
Using qword indices, the instruction conditionally loads up to 2 or 4 qword values from the VSIB addressing memory operand, and updates the lower half of the destination register. The upper 128 or 256 bits of the destination register are zero’ed with qword indices.
This instruction can be suspended by an exception if at least one element is already gathered (i.e., if the exception is triggered by an element other than the rightmost one with its mask bit set). When this happens, the destination register and the mask operand are partially updated; those elements that have been gathered are placed into the destination register and have their mask bits set to zero. If any traps or interrupts are pending from already gathered elements, they will be delivered in lieu of the exception; in this case, EFLAG.RF is set to one so an instruction breakpoint is not re-triggered when the instruction is continued.
If the data size and index size are different, part of the destination register and part of the mask register do not correspond to any elements being gathered. This instruction sets those parts to zero. It may do this to one or both of those registers even if the instruction triggers an exception, and even if the instruction triggers the exception before gathering any elements.
VEX.128 version: For dword indices, the instruction will gather four dword values. For qword indices, the instruction will gather two values and zero the upper 64 bits of the destination.
VEX.256 version: For dword indices, the instruction will gather eight dword values. For qword indices, the instruction will gather four values and zero the upper 128 bits of the destination.
Note that:
DEST := SRC1; BASE_ADDR: base register encoded in VSIB addressing; VINDEX: the vector index register encoded by VSIB addressing; SCALE: scale factor encoded by SIB:[7:6]; DISP: optional 1, 4 byte displacement; MASK := SRC3;
MASK[MAXVL-1:128] := 0; FOR j := 0 to 3 i := j * 32; IF MASK[31+i] THEN MASK[i +31:i] := FFFFFFFFH; // extend from most significant bit ELSE MASK[i +31:i] := 0; FI; ENDFOR FOR j := 0 to 3 i := j * 32; DATA_ADDR := BASE_ADDR + (SignExtend(VINDEX[i+31:i])*SCALE + DISP; IF MASK[31+i] THEN DEST[i +31:i] := FETCH_32BITS(DATA_ADDR); // a fault exits the instruction FI; MASK[i +31:i] := 0; ENDFOR DEST[MAXVL-1:128] := 0;
MASK[MAXVL-1:64] := 0; FOR j := 0 to 3 i := j * 32; IF MASK[31+i] THEN MASK[i +31:i] := FFFFFFFFH; // extend from most significant bit ELSE MASK[i +31:i] := 0; FI; ENDFOR FOR j := 0 to 1 k := j * 64; i := j * 32; DATA_ADDR := BASE_ADDR + (SignExtend(VINDEX1[k+63:k])*SCALE + DISP; IF MASK[31+i] THEN DEST[i +31:i] := FETCH_32BITS(DATA_ADDR); // a fault exits the instruction FI; MASK[i +31:i] := 0; ENDFOR DEST[MAXVL-1:64] := 0;
MASK[MAXVL-1:256] := 0; FOR j := 0 to 7 i := j * 32; IF MASK[31+i] THEN MASK[i +31:i] := FFFFFFFFH; // extend from most significant bit ELSE MASK[i +31:i] := 0; FI; ENDFOR FOR j := 0 to 7 i := j * 32; DATA_ADDR := BASE_ADDR + (SignExtend(VINDEX1[i+31:i])*SCALE + DISP; IF MASK[31+i] THEN DEST[i +31:i] := FETCH_32BITS(DATA_ADDR); // a fault exits the instruction FI; MASK[i +31:i] := 0; ENDFOR DEST[MAXVL-1:256] := 0;
MASK[MAXVL-1:128] := 0; FOR j := 0 to 7 i := j * 32; IF MASK[31+i] THEN MASK[i +31:i] := FFFFFFFFH; // extend from most significant bit ELSE MASK[i +31:i] := 0; FI; ENDFOR FOR j := 0 to 3 k := j * 64; i := j * 32; DATA_ADDR := BASE_ADDR + (SignExtend(VINDEX1[k+63:k])*SCALE + DISP; IF MASK[31+i] THEN DEST[i +31:i] := FETCH_32BITS(DATA_ADDR); // a fault exits the instruction FI; MASK[i +31:i] := 0; ENDFOR DEST[MAXVL-1:128] := 0;
VPGATHERDD: __m128i _mm_i32gather_epi32 (int const * base, __m128i index, const int scale);
VPGATHERDD: __m128i _mm_mask_i32gather_epi32 (__m128i src, int const * base, __m128i index, __m128i mask, const int scale);
VPGATHERDD: __m256i _mm256_i32gather_epi32 ( int const * base, __m256i index, const int scale);
VPGATHERDD: __m256i _mm256_mask_i32gather_epi32 (__m256i src, int const * base, __m256i index, __m256i mask, const int scale);
VPGATHERQD: __m128i _mm_i64gather_epi32 (int const * base, __m128i index, const int scale);
VPGATHERQD: __m128i _mm_mask_i64gather_epi32 (__m128i src, int const * base, __m128i index, __m128i mask, const int scale);
VPGATHERQD: __m128i _mm256_i64gather_epi32 (int const * base, __m256i index, const int scale);
VPGATHERQD: __m128i _mm256_mask_i64gather_epi32 (__m128i src, int const * base, __m256i index, __m128i mask, const int scale);
None.
See Table 2-27, “Type 12 Class Exception Conditions.”