窥孔优化
窥孔优化是一种优化编译器技术,在编译的后面阶段,寻找编译器生成的特定的指令序列,将该序列替换成性能更好的等效指令序列。该算法的可视范围极小,往往只能同时窥看几条指令,故名曰窥孔优化(指其犹如通过窥孔观察程序)。被优化掉的这少量指令序列被称为窥视孔或窗口。[1]
例如:
- 对于指令序列:将寄存器A压入堆栈,然后立即将值弹出存回寄存器A的指令序列,窥孔优化删除这两条指令。
- 对于指令A加到A上,窥孔优化替换为算术左移。
- 对于浮点寄存器乘以8,窥孔优化会将浮点寄存器的指数部分加上3。
- 对于数组寻址操作将索引乘以4,将结果与基地址相加以获得指针值,然后解引用该指针,窥孔优化可能会使用一种硬件寻址模式,通过一条指令实现相同的结果。
1965年William Marshall McKeeman提出窥孔优化一词。[2]
替换规则
窥孔优化中常用的技术:[3]
- 空序列——删除无用的操作
- 组合操作——用一项等价操作替换多项操作
- 代数定律 – 使用代数定律来简化或重新排序指令
- 特殊情况指令 – 使用为特殊操作数情况设计的指令
- 地址模式操作 – 使用地址模式来简化代码
还可以有其他类型的窥视孔优化。
例子
用较快的指令替换较慢的指令
以下为Java字节码
... aload 1 aload 1 mul ...
可以替换为
... aload 1 dup mul ...
与大多数窥孔优化一样,这种优化对指令的效率做出了某些假设。例如,在这种情况下,假设dup
操作(复制并推送到栈顶)比操作aload X
(加载标识为X
的局部变量并将其推送到栈上)更有效。
删除冗余代码
另一个例子是消除冗余load stores.
a = b + c; d = a + e;
直接实现为
MOV b, R0 ; Copy b to the register
ADD c, R0 ; Add c to the register, the register is now b+c
MOV R0, a ; Copy the register to a
MOV a, R0 ; Copy a to the register
ADD e, R0 ; Add e to the register, the register is now a+e [(b+c)+e]
MOV R0, d ; Copy the register to d
但可以优化为
MOV b, R0 ; Copy b to the register
ADD c, R0 ; Add c to the register, which is now b+c (a)
MOV R0, a ; Copy the register to a
ADD e, R0 ; Add e to the register, which is now b+c+e [(a)+e]
MOV R0, d ; Copy the register to d
删除冗余栈指令
如果编译器在调用子例程之前将寄存器保存在栈上并在返回时恢复它们,则对子例程的连续调用可能会产生冗余栈指令。
假设编译器为每个过程调用生成以下Z80指令:
PUSH AF
PUSH BC
PUSH DE
PUSH HL
CALL _ADDR
POP HL
POP DE
POP BC
POP AF
如果有两个连续的子例程调用,它们将如下所示:
PUSH AF
PUSH BC
PUSH DE
PUSH HL
CALL _ADDR1
POP HL
POP DE
POP BC
POP AF
PUSH AF
PUSH BC
PUSH DE
PUSH HL
CALL _ADDR2
POP HL
POP DE
POP BC
POP AF
对于相同的寄存器,POP regs 后跟 PUSH 的序列通常是多余的。在冗余的情况下,窥孔优化将删除这些指令。在示例中,这将导致另一个冗余的 POP/PUSH 对出现在窥视孔中,并且这些将依次被删除。假设子程序 _ADDR2 不依赖于先前的寄存器值,则删除上例中的所有冗余代码最终将留下以下代码:
PUSH AF
PUSH BC
PUSH DE
PUSH HL
CALL _ADDR1
CALL _ADDR2
POP HL
POP DE
POP BC
POP AF
实现
参见
参考文献
- ^ Muchnick, Steven Stanley. Advanced Compiler Design and Implementation. Academic Press / Morgan Kaufmann. 1997-08-15 [2023-08-22]. ISBN 978-1-55860-320-2. (原始内容存档于2023-08-22).
- ^ McKeeman, William Marshall. Peephole optimization. Communications of the ACM. July 1965, 8 (7): 443–444. S2CID 9529633. doi:10.1145/364995.365000 .
- ^ Fischer, Charles N.; Cytron, Ron K.; LeBlanc, Jr., Richard J. Crafting a Compiler (PDF). Addison-Wesley. 2010 [2018-07-02]. ISBN 978-0-13-606705-4. (原始内容 (PDF)存档于3 July 2018).
- ^ Aho, Alfred Vaino; Lam, Monica Sin-Ling; Sethi, Ravi; Ullman, Jeffrey David. Chapter 8.9.2 Code Generation by Tiling an Input Tree. Compilers – Principles, Techniques, & Tools (PDF) 2. Pearson Education. 2007: 540 [2018-07-02]. (原始内容存档 (PDF)于2018-06-10).