-
Notifications
You must be signed in to change notification settings - Fork 10
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Relaxed Integer Dot Product instructions #52
Comments
I'm in favor of trying to make this instruction work, but in reference to #9 i think we need to check whether that performance number is realizable when the more relaxed semantics are applied, or whether intermediate results need to be fixed up to account for the relaxed semantics. The experiment @kpu references was x86-only and the semantics were explicitly those of PMADDUBSW. @yurydelendik looked into whether the experiment could use the dot products on arm64 and ran into some trouble, iirc. |
FWIW the code that benefited from introducing PMADDUBSW can be found at https://github.com/kpu/intgemm/blob/d3657687c9b84d2ea2ea39b2fac7b89597bde848/intgemm/multiply.h#L305-L360 |
This instruction is the first one that is not just a relaxed version of SIMD proposal. I wonder if it should be |
During implementation I encountered this case PMADDUBS produces intermediate signed 16-bit integers, and the saturated result of adding is packed to the destination. The above 129 * 65 + 192 * 127 == 32769, but was saturated to 32767. |
@yurydelendik You are right, I missed the saturating behavior of |
The end goal is matrix multiply. Native matrix multiply implementations have highly-optimized implementations that are separate for x86 and ARM. If I'm honest about how I would use these instructions, I would:
Models quantized for the dominant case of native code will use all 8 bits and be prepared for Intel's saturation (some scale both arguments down by sqrt(2) to avoid saturation); restricting to 7 bits would require new models to be distributed and validated, probably with quantization loss. So really I'm asking for instructions whose behavior can be exploited to implement 8-bit multiply. |
What about just having an ARM USDOT / x86 VNNI wrapper that always does unsigned * signed? On pre-VNNI x86 it lowers to Disadvantage is older ARM with only SDOT (no USDOT) is slower than necessary. |
The specification of |
Updated the proposal to reflect the removal of |
Port commit a52b44f Original Commit Message: Prototype the instruction on the interpreter, and Arm64. Details of instruction lowerings on all relevant architectures can be found at: WebAssembly/relaxed-simd#52 Change-Id: Ie0415f5c6a543517aa488a36ea5e575c6612ec0e Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3687424 Auto-Submit: Yahan Lu <[email protected]> Commit-Queue: ji qiu <[email protected]> Reviewed-by: ji qiu <[email protected]> Cr-Commit-Position: refs/heads/main@{#80951}
For the Wasm SIMD 128 lowering, i assume the |
Yes, |
What are the instructions being proposed?
I propose relaxed 8-bit versions of the Dot Product instructions introduced in WebAssembly/simd#127. These instruction expose multipliplication of 8-bit (unsigned or signed) elements by 7-bit (treated as unsigned) elements with accumulation of adjacent products. These instructions are designed to expose the performance benefits of the following native instructions in a portable way:
PMADDUBSW
instructionAVX2-VNNI
orAVX512-VNNI
)VPDPBUSD
instruction.VSDOT.S8
and)VUDOT.U8
SDOT
and)UDOT
Discussion on Issue #9 goes into great length explaining the performance benefits of the above-mentioned native instructions.
I suggest
i16x8.dot_i8x16_i7x16_s
,,i16x8.dot_i8x16_i7x16_u
i32x4.dot_i8x16_i7x16_add_s
, andas the tentative names for the proposed instructions.i32x4.dot_i8x16_i7x16_add_u
What are the semantics of these instructions?
Both x86 and ARM provide variants of Dot Product instructions on SIMD vectors of 8-bit elements, but differ in the semantics of the input elements:
The proposed instructions resolve this incompatibility by guaranteeing the result only when elements of the second input SIMD vector have at most 7 non-zero bits, as in this case there is no difference between signed and unsigned representation.
i16x8.dot_i8x16_i7x16_s is a 2-element dot product instruction consuming signed 8-bit integer elements as the first input SIMD vector and 7-bit integer elements (treated as unsigned) as the second input SIMD vector and producing signed 16-bit integer output elements. The 2-element dot product never overflows as the worst case outputs fit into signed 16-bit integer:
-128
*127
+-128
*127
=-32512
>INT16_MIN
=-32768
127
*127
+127
*127
=32512
<INT16_MAX
=32767
i16x8.dot_i8x16_i7x16_u is a 2-element dot product instruction consuming unsigned 8-bit integer elements as the first input SIMD vector and 7-bit integer elements (treated as unsigned) as the second input SIMD vector and producing unsigned 16-bit integer output elements. The 2-element dot product never overflows as the worst case outputs fit into unsigned 16-bit integer:255
*127
+255
*127
=64770
<UINT16_MAX
=65536
i32x4.dot_i8x16_i7x16_add_s is a 4-element dot product with accumulation instruction consuming signed 8-bit integer elements in the first input SIMD vector, 7-bit integer elements (treated as unsigned) in the second input SIMD vector, and 32-bit integer elements (signedness-agnostic) in the third input SIMD vector and producing (signedness-agnostic) 32-bit integer output elements. The 4-element dot product producing a 32-bit result never overflows, and the addition of the third input SIMD vector is performed in modulo arithmetics.
i32x4.dot_i8x16_i7x16_add_u is a 4-element dot product with accumulation instruction consuming unsigned 8-bit integer elements in the first input SIMD vector, 7-bit integer elements (treated as unsigned) in the second input SIMD vector, and 32-bit integer elements (signedness-agnostic) in the third input SIMD vector and producing (signedness-agnostic) 32-bit integer output elements. The 4-element dot product producing a 32-bit result never overflows, and the addition of the third input SIMD vector is performed in modulo arithmetics.How will these instructions be implemented?
x86/x86-64 processors with AVX2-VNNI or AVX512-VNNI instruction set
i32x4.dot_i8x16_i7x16_add_s
c = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered toVPDPBUSD xmm_c, xmm_a, xmm_b
y = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered toVMOVDQA xmm_y, xmm_c
+VPDPBUSD xmm_c, xmm_a, xmm_b
i32x4.dot_i8x16_i7x16_add_uc = i32x4.dot_i8x16_i7x16_add_u(a, b, c)
is lowered toVPDPBUSD xmm_c, xmm_b, xmm_a
y = i32x4.dot_i8x16_i7x16_add_u(a, b, c)
is lowered toVMOVDQA xmm_y, xmm_c
+VPDPBUSD xmm_c, xmm_b, xmm_a
x86/x86-64 processors with XOP instruction set
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b)
is lowered to:VPMADDUBSW xmm_tmp, xmm_b, xmm_a
VPHADDWD xmm_tmp, xmm_tmp
VPADDD xmm_y, xmm_y, xmm_tmp
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_u(a, b)
is lowered to:VPMADDUBSW xmm_tmp, xmm_a, xmm_b
VPHADDWD xmm_tmp, xmm_tmp
VPADDD xmm_y, xmm_y, xmm_tmp
x86/x86-64 processors with AVX instruction set
i16x8.dot_i8x16_i7x16_s
y = i16x8.dot_i8x16_i7x16_s(a, b)
is lowered toVPMADDUBSW xmm_y, xmm_b, xmm_a
i16x8.dot_i8x16_i7x16_uy = i16x8.dot_i8x16_i7x16_u(a, b)
is lowered toVPMADDUBSW xmm_y, xmm_a, xmm_b
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b)
is lowered to:VPMADDUBSW xmm_tmp, xmm_b, xmm_a
VPMADDWD xmm_tmp, xmm_tmp, [wasm_i16x8_splat(1)]
VPADDD xmm_y, xmm_y, xmm_tmp
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_u(a, b)
is lowered to:VPMADDUBSW xmm_tmp, xmm_a, xmm_b
VPMADDWD xmm_tmp, xmm_tmp, [wasm_i16x8_splat(1)]
VPADDD xmm_y, xmm_y, xmm_tmp
x86/x86-64 processors with SSSE3 instruction set
i16x8.dot_i8x16_i7x16_s
y = i16x8.dot_i8x16_i7x16_s(a, b)
(y
is NOTa
) is lowered toMOVDQA xmm_y, xmm_b
+PMADDUBSW xmm_y, xmm_a
i16x8.dot_i8x16_i7x16_uy = i16x8.dot_i8x16_i7x16_u(a, b)
(y
is NOTb
) is lowered toMOVDQA xmm_y, xmm_a
+PMADDUBSW xmm_y, xmm_b
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b)
is lowered to:MOVDQA xmm_tmp, xmm_b
PMADDUBSW xmm_tmp, xmm_a
PMADDWD xmm_tmp, [wasm_i16x8_splat(1)]
PADDD xmm_y, xmm_tmp
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_u(a, b)
is lowered to:MOVDQA xmm_tmp, xmm_a
PMADDUBSW xmm_tmp, xmm_b
PMADDWD xmm_tmp, [wasm_i16x8_splat(1)]
PADDD xmm_y, xmm_tmp
ARM64 processors with Dot Product extension
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered toMOV Vy.16B, Vc.16B
+SDOT Vy.4S, Va.16B, Vb.16B
c = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered toSDOT Vc.4S, Va.16B, Vb.16B
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_u(a, b, c)
is lowered toMOV Vy.16B, Vc.16B
+UDOT Vy.4S, Va.16B, Vb.16B
c = i32x4.dot_i8x16_i7x16_add_u(a, b, c)
is lowered toUDOT Vc.4S, Va.16B, Vb.16B
ARM64 processors
i16x8.dot_i8x16_i7x16_s
y = i16x8.dot_i8x16_i7x16_s(a, b)
is lowered to:SMULL Vy.8H, Va.8B, 4B.8B
SMULL2 Vtmp.8H, Va.16B, 4B.16B
ADDP Vy.8H, Vy.8H, Vtmp.8H
i16x8.dot_i8x16_i7x16_uy = i16x8.dot_i8x16_i7x16_u(a, b)
is lowered to:UMULL Vy.8H, Va.8B, 4B.8B
UMULL2 Vtmp.8H, Va.16B, 4B.16B
ADDP Vy.8H, Vy.8H, Vtmp.8H
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered to:SMULL Vtmp.8H, Va.8B, 4B.8B
SMULL2 Vtmp2.8H, Va.16B, 4B.16B
ADDP Vtmp.8H, Vtmp.8H, Vtmp2.8H
SADDLP Vtmp.4S, Vtmp.8H
ADD Vy.4S, Vtmp.4S, Vc.8H
c = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered to:SMULL Vtmp.8H, Va.8B, 4B.8B
SMULL2 Vtmp2.8H, Va.16B, 4B.16B
ADDP Vtmp.8H, Vtmp.8H, Vtmp2.8H
SADALP Vc.4S, Vtmp.8H
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered to:UMULL Vtmp.8H, Va.8B, 4B.8B
UMULL2 Vtmp2.8H, Va.16B, 4B.16B
ADDP Vtmp.8H, Vtmp.8H, Vtmp2.8H
UADDLP Vtmp.4S, Vtmp.8H
ADD Vy.4S, Vtmp.4S, Vc.8H
c = i32x4.dot_i8x16_i7x16_add_u(a, b, c)
is lowered to:UMULL Vtmp.8H, Va.8B, 4B.8B
UMULL2 Vtmp2.8H, Va.16B, 4B.16B
ADDP Vtmp.8H, Vtmp.8H, Vtmp2.8H
UADALP Vc.4S, Vtmp.8H
Reference lowering through the WAsm SIMD128 instruction set
i16x8.dot_i8x16_i7x16_s
y = i16x8.dot_i8x16_i7x16_s(a, b)
is lowered to:const v128_t prod_low = wasm_i16x8_extmul_low_i8x16(a, b)
const v128_t prod_high = wasm_i16x8_extmul_high_i8x16(a, b)
const v128_t prod_even = wasm_v16x8_shuffle(prod_low, prod_high, 0, 2, 4, 6, 8, 10, 12, 14)
const v128_t prod_odd = wasm_v16x8_shuffle(prod_low, prod_high, 1, 3, 5, 7, 8, 11, 13, 15)
y = wasm_i16x8_add(prod_even, prod_odd)
i16x8.dot_i8x16_i7x16_uy = i16x8.dot_i8x16_i7x16_u(a, b)
is lowered to:const v128_t prod_low = wasm_u16x8_extmul_low_u8x16(a, b)
const v128_t prod_high = wasm_u16x8_extmul_high_u8x16(a, b)
const v128_t prod_even = wasm_v16x8_shuffle(prod_low, prod_high, 0, 2, 4, 6, 8, 10, 12, 14)
const v128_t prod_odd = wasm_v16x8_shuffle(prod_low, prod_high, 1, 3, 5, 7, 8, 11, 13, 15)
y = wasm_i16x8_add(prod_even, prod_odd)
i32x4.dot_i8x16_i7x16_add_s
y = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered to:const v128_t prod_low = wasm_i16x8_extmul_low_i8x16(a, b)
const v128_t prod_high = wasm_i16x8_extmul_high_i8x16(a, b)
const v128_t psum_low = wasm_i32x4_extadd_pairwise_i16x8(prod_low)
const v128_t psum_high = wasm_i32x4_extadd_pairwise_i16x8(prod_high)
const v128_t psum_even = wasm_v32x4_shuffle(psum_low, psum_high, 0, 2, 4, 6)
const v128_t psum_odd = wasm_v32x4_shuffle(psum_low, psum_high, 1, 3, 5, 7)
const v128_t psum = wasm_i32x4_add(prod_even, prod_odd)
y = wasm_i32x4_add(psum, c)
i32x4.dot_i8x16_i7x16_add_uy = i32x4.dot_i8x16_i7x16_add_s(a, b, c)
is lowered to:const v128_t prod_low = wasm_u16x8_extmul_low_u8x16(a, b)
const v128_t prod_high = wasm_u16x8_extmul_high_u8x16(a, b)
const v128_t psum_low = wasm_u32x4_extadd_pairwise_u16x8(prod_low)
const v128_t psum_high = wasm_u32x4_extadd_pairwise_u16x8(prod_high)
const v128_t psum_even = wasm_v32x4_shuffle(psum_low, psum_high, 0, 2, 4, 6)
const v128_t psum_odd = wasm_v32x4_shuffle(psum_low, psum_high, 1, 3, 5, 7)
const v128_t psum = wasm_i32x4_add(prod_even, prod_odd)
y = wasm_i32x4_add(psum, c)
How does behavior differ across processors? What new fingerprinting surfaces will be exposed?
As the native equivalents of the proposed Dot Product instructions on x86 perform signed-by-unsigned multiplication and the native equivalents on ARM perform either signed-by-signed, or unsigned-by-unsigned multiplications, it is possible to distinguish these architectures from results on out-of-bounds (when the high bit of the elements in the second input SIMD vector) inputs. x86/x86-64 can already be distinguished from ARM/ARM64 based on NaN behavior, so this aspect doesn't expose any new fingerprinting surfaces.
However, it is also possible to distinguish processors with AVX2-VNNI or AVX512-VNNI instruction sets on x86 from processors without these instruction sets by detecting saturation of intermediate results (in
PMADDUBSW
instruction), and distinguish ARM processors with Dot Product extension from ARM processors without this extension by detecting wrapping of intermediate results (inADDP
instructions). WebAssembly engines have three options to manage exposure of this fingerprinting surface:Wait it out, as new processors tend to support the AVX2-VNNI / AVX512-VNNI extension on the x86 and the NEON Dot Product extension on ARM. 2022 processor cores from Intel (Golden Cove, Gracemont), AMD (Zen 4), ARM (Cortex-X2, Cortex-A710, Cortex-A510), and Apple (A15) all support these instruction set extensions.
Mimick the behaviour of the AVX2-VNNI / AVX512-VNNI
VPDPBUSD
instruction on the x86 processors without this instruction extension and the behaviour of the NEON Dot Product instructions on the ARM processors without this instruction set extension. This option comes at a performance cost on the older processors.Avoid the AVX2-VNNI / AVX512-VNNI
VPDPBUSD
instruction on x86 and the NEON Dot Product instructions on ARM, and use the same instruction sequences as the older processors. This option comes at a performance cost on the newer processors.What use cases are there?
PMADDUBSW
instruction accelerates machine translation application compiled to WebAssembly by 310% over pure WebAssembly SIMD.The text was updated successfully, but these errors were encountered: