- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
这个问题是关于我们如何将一个整数与一个常数相乘。那么让我们看一个简单的函数:
int f(int x) {
return 10*x;
}
如何才能最好地优化该函数,尤其是在内联到调用方时?
方法 1(由大多数优化编译器生成(例如 on Godbolt))
lea (%rdi,%rdi,4), %eax
add %eax, %eax
方法 2(使用 clang3.6 及更早版本生成,使用 -O3)
imul $10, %edi, %eax
方法 3(使用 g++6.2 生成,未优化,删除存储/重新加载)
mov %edi, %eax
sal $2, %eax
add %edi, %eax
add %eax, %eax
哪个版本最快,为什么?主要对 Intel Haswell 感兴趣。
最佳答案
根据 Agner Fog's testing (以及 AIDA64 等其他东西)自 Core2 以来的英特尔 CPU 的 imul r32,r32, imm
延迟为 3c,吞吐量为每 1c。自 Nehalem 以来,64 位乘法也那么快。 (Agner 说 Nehalem 的 imul r64,r64,imm
比 imul r64,r64
慢(2c 吞吐量),但这与其他结果不匹配。Instlatx64 says 1c. )
Ryzen 之前的 AMD CPU 速度较慢,例如对于 32 位乘法,Steamroller 有 lat=4c tput=one per 2c。对于 64 位乘法,lat=6c tput=one per 4c。 AMD Ryzen 具有与 Intel 一样出色的乘法性能。
在寻址模式下具有 2 个组件的 LEA(基址 + 索引,但没有恒定位移)在所有 Intel CPU 上以 1c 延迟运行1,除了 Atom,LEA 在不同阶段运行流水线(在实际的 AGU 中,而不是 ALU)并且需要比“正常”ALU 指令早 4c 就绪的输入。相反,我认为它的输入会更快准备好,因此 ADD 可以使用相同周期的结果。 (我没有对此进行测试,也没有任何 Atom 硬件。)
在 Intel SnB 系列上,simple-LEA 可以在端口 1 或 5 上运行,因此它的吞吐量是 IMUL 的两倍。
ADD 可以在任何 CPU 上的任何 ALU 端口上运行。 HSW 引入了第 4 个 ALU 端口(相对于 IvyBridge),因此它可以维持每个时钟 4 个 ALU 微指令(理论上)。
所以 LEA+ADD 版本在大多数 x86 CPU 上有 2c 延迟,而在 Haswell 上每个时钟可以运行两次乘法。
脚注 1:在 AMD(包括 Zen/Zen2)上,缩放索引使 LEA“变慢”(2 个周期延迟并在更少的端口上运行)。例如lea r32,[r64+r64*2]
在 2 个周期延迟下测得 on Zen2与 1 个周期 on Skylake . (Agner Fog 还提到 lea r32, [r64...]
在 AMD 上速度较慢,但这可能只是推土机效应;在 Zen/Zen2 的 https://uops.info/'s 结果中并不明显。 )
但是,如果乘法只是更大的周围循环的一小部分,它会限制总 uop 吞吐量,而不是乘法延迟或吞吐量,则 IMUL 版本更好。
如果您的乘法常数对于两个 LEA 或 SHL+LEA 来说太大,那么您可能最好使用 IMUL,尤其是在主要针对具有极高性能整数乘法器的 Intel CPU 进行调整时。
SHL+LEA 或 SHL+SUB 可能有用,例如乘以 63。(来自 Godbolt:gcc6.2 -O3 -march=haswell
)
movl %edi, %eax
sall $6, %eax
subl %edi, %eax
在 Haswell 上,MOV 是零延迟的,这只有 2c 延迟。但对于 imull $63, %edi, %eax
,它是 3 个融合域 uops 而不是 1 个。所以它在管道中有更多的微指令,减少了 CPU 可以“看到”进行乱序执行的超前距离。它还增加了 uop 缓存和 L1 I 缓存的压力,因为它需要更多的指令字节,因此编译器会始终选择这种策略。
在 IvyBridge 之前的 CPU 上,这绝对比 IMUL 差,除非有其他东西在竞争端口 1,因为它是 3c 延迟(MOV 在关键路径依赖链上,并且有 1c 延迟)。
对于不同微架构上的相同周边代码,答案也会不同。
关于assembly - 常量乘法 - imul 或 shl-add-combination,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40564004/
我想使用 Haskell 的 parsec 库来实现这个语法规则: ((a | b | c)* (a | b))? 这是一个接受可选(即可能为空)字符串的解析器规则。如果它接受的字符串不为空,则可以通
Python 的 itertools.combinations() 创建的结果是数字的组合。例如: a = [7, 5, 5, 4] b = list(itertools.combinations(a
I found a good script for the permutation of lists, combining and not combining list position on
我正在使用 Beam 管道计算流式数据的电话号码频率。我使用的滑动窗口每 5 分钟重复一次,总周期为 15 分钟,因此正如预期的那样,对于某些输入,当输入落在多个窗口中时,我会得到多个输出。 计算出现
这个问题已经有答案了: Pandas Merging 101 (8 个回答) 已关闭 3 年前。 我有两个数据帧,我想对其执行外连接。两个数据框共享一个公共(public)索引名称以及多个也共享相同名
我在谷歌上搜索了很多天这个问题,但一无所获。我需要做一个 SELECT,DUPLICATE 和 DUPLICATE 和 DUPLICATE 取决于用户。之后,我需要将每个选项的值组合到我选择的一个选择
这个问题在这里已经有了答案: Java 8 Streams: multiple filters vs. complex condition (4 个答案) 关闭 4 年前。 需要过滤所有适合其领域某
运行 cv2.getRectSubPix(img, (5,5), (0,0)) 抛出错误: OpenCV Error: Unsupported format or combination of for
没有重复的组合看起来像这样,当可供选择的元素数 (n) 为 5 且选择的元素数 (r) 为 3 时: 0 1 2 0 1 3 0 1 4 0 2 3 0 2 4 0 3 4 1 2 3 1 2 4 1
我在学校的数学一直不太好,我意识到我实际上需要与 pow(base, exponent) 函数相反的函数,该函数对某个数字进行乘方运算,例如 2 ^ 4 = 16 搜索答案我发现对数 log() 应该
我确信这很简单,但我很难找到一种方法来做到这一点。基本上,如果我有一个包含 P 列和 V^P 行的数组,我如何填写所有组合,即基本上所有可能的数字以 P 数字的 V 为基数。例如,对于 P=3 和 V
我想知道一种可能的算法来计算所有可能的组合,没有重复,从 length=1 开始,直到 length=N 的 N 个元素。 例子: 元素:1、2、3。 输出: 1 2 3 12 13 23 123 最
使用三种不同颜色的颜料可以用多少种不同的方式来绘制立方体? 最佳答案 如果您以唯一可能的有趣方式解释它,那么这是一个比 3^6 更难的问题:有多少种不同的(即对称的)方法来为立方体着色。这是一篇论文:
我正在尝试解决优化问题,但首先我必须找到 n 个元素的所有可能组合的数量,但要考虑一些冲突。一个可能的例子是: 元素:{1,2,3,4}冲突:{1,2},{3,4} 术语“冲突”是指属于同一冲突集合的
Cleave 是一个非常有用的组合器,可以最大限度地减少代码重复。假设我要分类 Abundant, Perfect, Deficient numbers : USING: arrays assocs
有没有办法让 @Published 变量只在新值与旧值不同时才发布其值? 现在如果我们有 @Published var test: Bool = false 我们做到了 test = false te
有一个数组 [1, 2, ..., m] ,并且有一个整数 n . 如 m=2和 n=3 ,我想获得 [1, 1, 1] [1, 1, 2] [1, 2, 1] [1, 2, 2] [2, 1, 1]
我在我的应用程序中使用了一个用于日志记录页面的表单,并且在页脚上有一个绑定(bind)来显示任何错误,如下所示: 内容 View .Swift : Form { Section(footer: Tex
HTML first second third SCSS $statistics: ("first", "second", "third"); :root { --first: r
我有一个 HTTP 请求发布者,当返回 401 错误时,我想停止执行并显示我的登录屏幕。 这是我的代码的一部分: cancellable = fetcher.hello(helloRequest: H
我是一名优秀的程序员,十分优秀!