- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试使用 simd(AVX2 或更早版本)查找字符的第一个实例,在本例中为 '"'。我想使用 _mm256_cmpeq_epi8,但随后我需要一种快速方法来查找是否存在以下任何一个: __m256i 中的结果字节已设置为 0xFF。然后计划使用 _mm256_movemask_epi8 将结果从字节转换为位,然后使用 ffs 获取匹配索引。使用一次移出一部分是否更好_mm_movemask_epi8?还有其他建议吗?
最佳答案
您的想法是正确的 _mm256_cmpeq_epi8
-> _mm256_movemask_epi8
。 AFAIK,至少对于 Intel CPU 来说这是实现这一点的最佳方法。 PMOVMSKB r32, ymm
与 XMM 16 字节版本的速度相同,因此解压 256b 向量的两个 channel 并分别 movemask 它们然后重新组合整数结果将是巨大的损失。 (来源: Agner Fog's instruction table 。请参阅 x86 标签 wiki 中的其他性能链接。)
通过保留ffs
,使循环内的代码尽可能高效。直到您从 _mm256_movemask_epi8
中识别出非零结果之后.
TEST/JCC 可以宏融合到单个微指令中,但 BSF/JCC 不能,因此需要额外的指令。 (无论如何,你都很难让 C 编译器发出 BSF/JCC。更有可能的是,对 ffs
的结果进行分支会给你某种输入非零的测试,然后是 BSF,然后加 1,然后比较并分支。与仅测试 movemask 结果相比,这显然很糟糕。)
(更新,在 C++20 中,使用 std::countr_zero
。它可以编译为单个 tzcnt
,而不是 ffs
的差一。因为您'已经检查过掩码是否非零,如果不确定运行该代码的所有 CPU 都支持 rep
,希望可以优化为单个 ( bsf
) tzcnt
指令。如果您可以假设 BMI1 在您的 objective-c PU(通常可用于 AVX2 代码),然后启用它,这样您就可以可靠地获得高效的 tzcnt
。)
另请注意,对于类似的问题,比较 movemask(例如检查它是否为 0xFFFFFFFF)与非零分支一样有效。
<小时/>正如 Paul R 所建议的,查看一些 strlen、strchr 和 memchr 实现可能会提供很多信息。在开源libc实现等地方有多种手写的asm实现。 (例如 glibc 和 Agner Fog's asmlib 。)
许多 glibc 的版本都会扫描到对齐边界,然后使用一次读取 64B 的展开循环(在 4 个 SSE 向量中,因为我认为 glibc 没有 AVX2 版本)。
要优化长字符串,请通过将比较结果进行“或”运算并进行检查来减少测试比较结果的开销。如果您发现命中,请返回并重新测试您的向量以查看哪个向量命中。
执行 ffs
可能会更有效基于您根据多个 movemask 结果构建的一个 64 位整数(使用 shift 和 |
)。我不确定在测试零之前是否在循环内执行此操作;我不记得 glibc 的 strlen 策略之一是否做到了这一点。
我在这里建议的所有内容都可以在 asm 中的 strlen、memchr 和相关函数的各种 glibc 策略中看到。这是sysdeps/x86_64/strlen.S ,但我可能在某个地方有另一个源文件使用了超过基线的 SSE2。 (或者不是,我可能正在考虑一个不同的函数,也许除了 SSE2 之外没有什么可以得到的,直到 AVX(3 操作数 insns)和 AVX2(256b 整数向量)。
另请参阅:
strchr-avx2.S
(Woboq.org 有一个很好的源浏览器,可以对文件名/符号进行有用的搜索)。memchr-avx2.S
glibc's memchr使用 PMAXUB 代替 POR。我不确定这对于某些神秘的微架构原因是否有用,但它在大多数 CPU 上运行在较少的端口上。也许这是所希望的,以避免与其他东西发生资源冲突? IDK,看起来很奇怪,因为它与 PCMPEQB 竞争。
关于x86 - 使用 simd 查找字符的第一个实例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40915243/
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 7年前关闭。 Improve this qu
我有一个代码库,我可以在我的 mac 上编译和运行,但不能在我的远程 linux 机器上编译和运行,我不确定为什么。 编译时出现错误 fatal error: simd/simd.h: No such
我需要了解如何编写一些可并行化问题的 C++ 跨平台实现,以便在可用的情况下利用 SIMD(SSE、SPU 等)。以及我希望能够在运行时在 SIMD 和非 SIMD 之间切换。 您建议我如何解决这个问
我正在使用 AVX 内在 _mm256_extract_epi32() . 不过,我不完全确定我是否正确使用它,因为 gcc 不喜欢我的代码,而 clang 编译它并运行它没有问题。 我根据整数变量的
当我可以使用 SSE3 或 AVX 时,SSE2 或 MMX 等较旧的 SSE 版本是否可用 - 还是我还需要单独检查它们? 最佳答案 一般来说,这些都是附加的,但请记住,多年来英特尔和 AMD 对这
在 godbolt.org 使用 gcc 7.2 我可以看到以下内容 code在汇编程序中翻译得非常好。我看到 1 次加载、1 次添加和 1 次存储。 #include __attribute__(
假设我们有一个函数将两个数组相乘,每个数组有 1000000 个 double 值。在 C/C++ 中,该函数如下所示: void mul_c(double* a, double* b) {
我有一个 A = a1 a2 a3 a4 b1 b2 b3 b4 c1 c2 c3 c4 d1 d2 d3 d4 我有两排, float32x2_t a = a1 a2 flo
我正在考虑编写一个 SIMD vector 数学库,因此作为一个快速基准,我编写了一个程序,该程序执行 1 亿(4 个 float ) vector 元素乘法并将它们加到累积总数中。对于我的经典非 S
我正在开发带有英特尔编译器 OpenMP 4.0 的英特尔 E5(6 核、12 线程) 为什么这段代码 SIMD 编译比并行 SIMD 编译更快? for (int suppv = 0; suppv
OpenMP 4.0 引入了 SIMD 结构以利用 CPU 的 SIMD 指令。根据规范http://www.openmp.org/mp-documents/OpenMP4.0.0.pdf ,有两种结
英特尔编译器允许我们通过以下方式对循环进行矢量化 #pragma simd for ( ... ) 但是,您也可以选择使用 OpenMP 4 的指令执行此操作: #pragma omp simd fo
关注我的 x86 question ,我想知道如何在 Arm-v8 上有效地矢量化以下代码: static inline uint64_t Compress8x7bit(uint64_t x) {
Intel 提供了几个 SIMD 命令,它们似乎都对 128 位数据执行按位异或: _mm_xor_pd(__m128d, __m128d) _mm_xor_ps(__m128, __m128) _m
可以使用“位打包”技术压缩无符号整数:在一个无符号整数 block 中,只存储有效位,当一个 block 中的所有整数都“小”时,会导致数据压缩。该方法称为 FOR (引用框架)。 有SIMD lib
SSE 寄存器是否在逻辑处理器(超线程)之间共享或复制? 对于 SSE 繁重的程序,我能否期望从并行化中获得与普通程序相同的加速(英特尔声称具有超线程的处理器为 30%)? 最佳答案 从英特尔的文档中
我正在编写一个使用 SSE 指令来乘法和相加整数值的程序。我用浮点数做了同样的程序,但我的整数版本缺少一个指令。 使用浮点数,在完成所有操作后,我将 de 值返回到常规浮点数数组,执行以下操作: _m
我正在开发基于Intel指令集(AVX,FMA等)的高性能算法。当数据按顺序存储时,我的算法(内核)运行良好。但是,现在我面临一个大问题,但没有找到解决方法或解决方案: see 2D Matrix i
大家好 :) 我正在尝试了解有关浮点、SIMD/数学内在函数和 gcc 的快速数学标志的一些概念。更具体地说,我在 x86 cpu 上使用 MinGW 和 gcc v4.5.0。 我已经搜索了一段时间
根据https://sourceware.org/glibc/wiki/libmvec GCC 具有数学函数的向量实现。它们可以被编译器用于优化,可以在这个例子中看到:https://godbolt.
我是一名优秀的程序员,十分优秀!