Guy*_*y B 5 x86 simd avx micro-optimization avx2
我需要使用LZCNT对字数组进行反向扫描:16位。
在Intel最新一代处理器上,LZCNT的吞吐量是每个时钟1次执行。AMD Ryzen的吞吐量似乎是4。
我试图找到一种使用AVX2指令集的算法来更快。
我知道AVX-512具有VPLZCNTD为32位元素,所以如果我有AVX512CD我可以解压并使用它。
仅使用AVX2指令集,就可以比使用x86 asm LZCNT指令更快地编码算法。
#include <immintrin.h>
__m256i avx2_lzcnt_epi16(__m256i v) {
const __m256i lut_lo = _mm256_set_epi8(
4, 4, 4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 6, 6, 7, 16,
4, 4, 4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 6, 6, 7, 16
);
const __m256i lut_hi = _mm256_set_epi8(
0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 3, 16,
0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 3, 16
);
const __m256i nibble_mask = _mm256_set1_epi8(0x0F);
const __m256i byte_offset = _mm256_set1_epi16(0x0008);
__m256i t;
t = _mm256_and_si256(nibble_mask, v);
v = _mm256_and_si256(_mm256_srli_epi16(v, 4), nibble_mask);
t = _mm256_shuffle_epi8(lut_lo, t);
v = _mm256_shuffle_epi8(lut_hi, v);
v = _mm256_min_epu8(v, t);
t = _mm256_srli_epi16(v, 8);
v = _mm256_or_si256(v, byte_offset);
v = _mm256_min_epu8(v, t);
return v;
}
// 16 - lzcnt_u16(subwords)
__m256i avx2_ms1b_epi16(__m256i v) {
const __m256i lut_lo = _mm256_set_epi8(
12, 12, 12, 12, 12, 12, 12, 12, 11, 11, 11, 11, 10, 10, 9, 0,
12, 12, 12, 12, 12, 12, 12, 12, 11, 11, 11, 11, 10, 10, 9, 0
);
const __m256i lut_hi = _mm256_set_epi8(
16, 16, 16, 16, 16, 16, 16, 16, 15, 15, 15, 15, 14, 14, 13, 0,
16, 16, 16, 16, 16, 16, 16, 16, 15, 15, 15, 15, 14, 14, 13, 0
);
const __m256i nibble_mask = _mm256_set1_epi8(0x0F);
const __m256i adj = _mm256_set1_epi16(0x1F08);
__m256i t;
t = _mm256_and_si256(nibble_mask, v);
v = _mm256_and_si256(_mm256_srli_epi16(v, 4), nibble_mask);
t = _mm256_shuffle_epi8(lut_lo, t);
v = _mm256_shuffle_epi8(lut_hi, v);
v = _mm256_max_epu8(v, t);
t = _mm256_srli_epi16(v, 8);
v = _mm256_sub_epi8(v, adj);
v = _mm256_max_epi8(v, t);
return v;
}
Run Code Online (Sandbox Code Playgroud)
对于打包到uint8中的结果,请使用_mm256_packs_epi16()。对于打包结果,请使用正确的顺序_mm256_permute4x64_epi64()。
来自r / SIMD的解决方案。此解决方案也在此处的注释中进行了描述。