# 改进RISC-V的代码生成

廖春玉 chunyu@iscas.ac.cn

# 自我介绍

- Phabricator <a href="https://reviews.llvm.org/p/liaolucy/">https://reviews.llvm.org/p/liaolucy/</a>
- github <u>https://github.com/ChunyuLiao</u>

# 简介

- 仓库: <a href="https://github.com/llvm/llvm-project">https://github.com/llvm/llvm-project</a>
- review代码: <u>https://reviews.llvm.org</u>
- 各种讨论/通知: https://llvm.discourse.group
- 手册: <a href="https://llvm.org/docs/DeveloperPolicy.html">https://llvm.org/docs/DeveloperPolicy.html</a>
- 针对RISCV的代码实现:

https://github.com/llvm/llvm-project/tree/main/llvm/lib/Target/RISCV

#### 简介

RISCVInstrFormats.td

● 针对RISCV的代码实现:

https://github.com/llvm/llvm-project/tree/main/llvm/lib/Target/RISCV

AsmParser RISCVInstrFormatsV.td CMakeLists.txt RISCVInstrInfoA.td Disassembler RISCVInstrInfo.cpp GISe1 RISCVInstrInfoC.td MCA RISCVInstrInfoD.td **MCTargetDesc** RISCVInstrInfoF.td RISCVAsmPrinter.cpp RISCVInstrInfo.h RISCVCallingConv.td RISCVInstrInfoM.td RISCVCodeGenPrepare.cpp RISCVInstrInfo.td RISCVExpandAtomicPseudoInsts.cpp RISCVInstrInfoVPseudos.td RISCVExpandPseudoInsts.cpp RISCVInstrInfoVSDPatterns.td RISCVFeatures.td RISCVInstrInfoV.td RISCVFrameLowering.cpp RISCVInstrInfoVVLPatterns.td RISCVFrameLowering.h RISCVInstrInfoXSf.td RISCVGatherScatterLowering.cpp RISCVInstrInfoXTHead.td RISCV.h RISCVInstrInfoXVentana.td RISCVInsertNTLHInsts.cpp RISCVInstrInfoZb.td RISCVInsertVSETVLI.cpp RISCVInstrInfoZc.td RISCVInstrFormatsC.td RISCVInstrInfoZfa.td

RISCVInstrInfoZfh.td

RISCVInstrInfoZicho.td RISCVInstrInfoZicond.td RISCVInstrInfoZihintntl.td RISCVInstrInfoZk.td RISCVInstrInfoZvk.td RISCVISelDAGToDAG.cpp RISCVISelDAGToDAG.h RISCVISelLowering.cpp RISCVISelLowering.h RISCVMachineFunctionInfo.cpp RISCVMachineFunctionInfo.h RISCVMacroFusion.cpp RISCVMacroFusion.h RISCVMakeCompressible.cpp RISCVMCInstLower.cpp RISCVMergeBaseOffset.cpp RISCVOptWInstrs.cpp RISCVProcessors.td RISCVRedundantCopvElimination.cpp RISCVRegisterInfo.cpp

RISCVRegisterInfo.h RISCVRegisterInfo.td RISCVRVVInitUndef.cpp RISCVSchedRocket.td RISCVSchedSiFive7.td RISCVSchedSyntacoreSCR1.td RISCVSchedule.td RISCVScheduleV.td RISCVScheduleZb.td RISCVSubtarget.cpp RISCVSubtarget.h RISCVSystemOperands.td RISCVTargetMachine.cpp RISCVTargetMachine.h RISCVTargetObjectFile.cpp RISCVTargetObjectFile.h RISCVTargetTransformInfo.cpp RISCVTargetTransformInfo.h RISCV.td

TargetInfo

# 概述

- 个人发现的改进机会
- Craig Topper, RISC-V Sign Extension Optimizations
- 开发过程中遇到的一些问题/故事

# 改进机会

- 主要集中在指令选择和指令生成阶段
- 更好的使用RISCV ISA 和 psABI 规范
- 对比借鉴gcc的优化

#### 尽量使用0比较

- RISCV有针对0比较的特殊指令
  - beqz/bnez, It expands to beq rs1, x0, offset.
  - seqz/snez, SLTIU rd, rs1, 1 sets rd to 1 if rs1 equals zero, otherwise sets rd to 0 (assembler pseudoinstruction SEQZ rd, rs).

 [RISCV] Enable preferZeroCompareBranch to optimize branch on zero in codegenprepare: <a href="https://reviews.llvm.org/D142071">https://reviews.llvm.org/D142071</a>

#### imm立即数优化

 [RISCV]Preserve (and X, 0xffff) in targetShrinkDemandedConstant <a href="https://reviews.llvm.org/D134155">https://reviews.llvm.org/D134155</a>

• 对比 https://llvm.godbolt.org/z/d1a9oTGcM

#### 减少分支指令数

- [RISCV] Branchless lowering for (select (x < 0), TrueConstant,</li>
  FalseConstant) and (select (x >= 0), TrueConstant, FalseConstant)
  <a href="https://reviews.llvm.org/D137949">https://reviews.llvm.org/D137949</a>
- [RISCV]Keep (select c, 0/-1, X) during PerformDAGCombine https://reviews.llvm.org/D139272

对比: <u>https://llvm.godbolt.org/z/rYezq7743</u>

#### tail-call 优化

- RISCV psABI <a href="https://github.com/riscv-non-isa/riscv-elf-psabi-doc">https://github.com/riscv-non-isa/riscv-elf-psabi-doc</a>
- [RISCV] Permit tail call to an externally-defined function with weak linkage <a href="https://reviews.llvm.org/D143137">https://reviews.llvm.org/D143137</a>

推荐大佬们的blog, 比如MaskRay的blog <a href="https://maskray.me/blog/2021-04-25-weak-symbol">https://maskray.me/blog/2021-04-25-weak-symbol</a>

# 钩子函数

- [RISCV]Enable isIntDivCheap when attribute is minsize <a href="https://reviews.llvm.org/D130543">https://reviews.llvm.org/D130543</a>
- [RISCV] Return false from shouldFormOverflowOp when type is i8 and i16 <a href="https://reviews.llvm.org/D143646">https://reviews.llvm.org/D143646</a>

对比: <u>https://llvm.godbolt.org/z/c5Toh6TEa</u>

#### 常见/其他架构已有优化

- [RISCV] Add more patterns for FNMADD <a href="https://reviews.llvm.org/D126852">https://reviews.llvm.org/D126852</a>
- [RISCV] Optimize 2x SELECT for floating-point types
- https://reviews.llvm.org/D127871

### 针对RV32的优化

• [LegalizeTypes][RISCV] Add a special case for (add X, -1) to ExpandIntRes\_ADDSUB <a href="https://reviews.llvm.org/D146635">https://reviews.llvm.org/D146635</a>

# 捡漏

 [RISCV] Use hasAllWUsers to recover XORI/ORI <a href="https://reviews.llvm.org/D135538">https://reviews.llvm.org/D135538</a>

# RISC-V Sign Extension Optimizations

 https://llvm.org/devmtg/2022-11/slides/TechTalk21-RISC-VSignExtensionOpti mizations.pdf

#### 开发过程中遇到的一些问题/故事

- 兼顾不同架构
  - [SelectionDAG][RISCV][X86][AArch64][AMDGPU][PowerPC] Improve SimplifyDemandedBits for SHL with NUW/NSW flags. <a href="https://reviews.llvm.org/D140665">https://reviews.llvm.org/D140665</a>
- 兼顾不同模块
  - [CodeGenPrepare][RISCV] Reverse transform in CGP to use zero-compare branch <a href="https://reviews.llvm.org/D147789">https://reviews.llvm.org/D147789</a>
- 兼顾不同的代码形似
  - [DAGCombiner][VP] Add DAGCombine for VP\_MUL. <a href="https://reviews.llvm.org/D121187">https://reviews.llvm.org/D121187</a>

# 开发过程中遇到的一些问题/故事

- 虽然有一些困难, 但是也有很大收获
- 大佬与我们同在

 [LegalizeTypes][RISCV] Add a special case to ExpandIntRes\_UADDSUBO for (uaddo X, 1). <a href="https://reviews.llvm.org/D144614">https://reviews.llvm.org/D144614</a>

# 欢迎交流,一起发掘更多优化机会