10 KiB
Lab2 实验记录:中间表示生成
1. 实验目标
本次 Lab2 的目标是在已有的 SysY 前端基础上,补齐语义检查与 IR 生成流程,使编译器能够把更完整的 SysY 程序翻译为 LLVM 风格 IR,并通过 llc/clang 验证生成结果的正确性。
本次完成工作的重点包括:
- 扩展 IR 类型系统与指令系统,支持
float、数组、分支、函数调用、GEP、类型转换等基础能力。 - 扩展 Sema,支持嵌套作用域、左值绑定、函数调用绑定与内建库函数预声明。
- 完成表达式、控制流、函数、数组与全局变量的 IR 生成逻辑。
- 修复全局初始化常量求值、短路求值、数组寻址、IR 打印格式等会直接阻塞 Lab2 验证的关键问题。
2. 代码改动范围
本次实验主要修改了以下模块:
src/sem与include/semsrc/ir与include/irsrc/irgen与include/irgen- 新增本文档
doc/Lab2-实验记录.md
其中:
sem负责名称绑定、作用域和语义信息准备。ir负责 IR 基础设施、Builder 与 Printer。irgen负责把 ANTLR 语法树翻译成 IR。
3. 完成过程
3.1 先确认问题边界
开始时先阅读了实验文档 doc/Lab2-中间表示生成.md,然后检查了以下实现:
IRGenDecl.cppIRGenExp.cppIRGenStmt.cppIRGenFunc.cppIRBuilder.cppIRPrinter.cppSema.cpp
在初始状态下,代码已经完成了大部分 Lab2 框架,但仍存在两个会直接导致失败的问题:
- 全局常量初始化时,
EvalConstExpr实际上仍然调用了运行时的EvalExpr,从而在没有插入点时进入builder_.CreateLoad/CreateBinary/...,最终报错:IRBuilder 未设置插入点 - 数组相关的指针/聚合类型处理不一致,局部数组、多维数组与数组参数传递时很容易触发
LoadInst 不支持的指针类型或生成错误的 GEP。
为了避免只靠静态阅读猜问题,随后先执行了构建与最小样例验证,确认真实失败点。
3.2 建立回归基线
首先重新构建项目:
cmake -S . -B build -DCMAKE_BUILD_TYPE=Release
cmake --build build -j 4
然后针对典型样例做验证:
simple_add.sy05_arr_defn4.sy95_float.sy
结果表明:
95_float.sy会因为全局常量路径错误触发IRBuilder 未设置插入点05_arr_defn4.sy会因为数组寻址/存储类型不一致导致崩溃
这一步的作用是把问题从“感觉哪里有问题”缩小到“常量求值路径”和“数组存储/寻址路径”两条主线。
4. 关键困难与解决办法
4.1 困难一:全局初始化错误地走了运行时 IRBuilder 路径
现象
像下面这种代码在全局或常量初始化中会崩溃:
const float PI = 3.1415926;
const int A = 1 + 2;
原因是原来的 EvalConstExpr 虽然名字叫“常量求值”,但内部还是直接调用了 EvalExpr。一旦表达式中包含需要访问变量、二元运算、短路逻辑等节点,就会落入 builder_ 创建指令的逻辑,而此时全局作用域没有任何基本块插入点。
解决办法
把编译期常量求值彻底独立出来:
- 为
EvalConstExpr单独实现一套常量求值 Visitor。 - 常量路径只返回
ConstantInt/ConstantFloat,绝不生成 IR 指令。 - 支持:
- 整数/浮点字面量
- 括号表达式
+、-、!* / % + -- 比较运算
&& ||- 标量
const的引用
- 在全局和常量初始化中,只允许使用
EvalConstExpr的结果。
效果
修复后:
- 全局初始化不再依赖插入点
95_float.sy中的全局常量能够稳定生成- 短路表达式在常量上下文中只做纯编译期求值,不会试图分配
alloca
4.2 困难二:数组变量、数组参数与标量变量的“存储语义”混乱
现象
原实现里,alloca/load/store/GEP 对类型的理解不统一:
- 标量变量需要的是“指向标量的槽位”
- 局部数组需要的是“聚合对象的基址”
- 数组形参在 SysY 中本质上是指针,不应按局部数组同样处理
如果把这些情况混在一起,就会出现:
load试图从数组类型直接取值- GEP 基类型和索引序列不匹配
- 局部数组访问、多维数组访问、数组实参传递行为错误
解决办法
做了三层拆分:
-
标量与数组分离
- 标量局部变量使用真正的标量槽位:
i32*或float* - 数组局部变量保留聚合基址
- 标量局部变量使用真正的标量槽位:
-
普通数组与数组形参分离
- 局部/全局数组通过多级 GEP 沿数组维度寻址
- 数组形参按“指针退化”处理,访问时根据剩余维度计算偏移
-
左值取址与值求值分离
GetLValuePtr只负责拿地址visitLValueExp根据左值是否仍是数组来决定是load还是数组退化传参
效果
修复后:
simple_add.sy恢复正常05_arr_defn4.sy可以生成并运行- 多维数组和数组形参的寻址逻辑更加稳定
4.3 困难三:局部数组花括号初始化语义不正确
现象
05_arr_defn4.sy 虽然在中期已经不再崩溃,但运行结果仍然错误,退出码从预期的 21 变成了 13。这说明不是寻址崩了,而是初始化布局错了。
问题根源在于:
- 一部分初始化按“子数组递进”处理
- 一部分初始化又按“标量扁平展开”处理
两套逻辑混用后,多维数组初始化次序就会乱掉。
解决办法
把局部数组初始化统一改成“聚合初始化 + 标量游标”方案:
- 先统一做零初始化
- 再对花括号初始化维护一个标量游标
- 标量初始化时按当前扁平偏移定位到实际元素
- 子聚合初始化时按当前对齐边界进入对应子数组
这套逻辑与 SysY/LLVM 前端常见的聚合初始化处理方式更接近。
效果
修复后 05_arr_defn4.sy 的 IR 可以通过 verify_ir.sh --run,输出与预期一致。
4.4 困难四:IR 文本虽然能打印,但 LLVM 后端不一定接受
现象
在进入 verify_ir.sh 阶段后,又暴露出一批“IR 生成没崩,但 LLVM 不认”的问题:
- 内建函数被打印成了空定义,而不是声明
- 浮点常量打印格式不符合 LLVM 期望
icmp/fcmp的结果在打印和后续使用中对i1/i32处理不一致- 自动临时名使用纯数字,打乱后会违反 LLVM 的编号要求
- 基本块名重复
getelementptr打印时的基类型信息不正确
解决办法
对 IR 基础设施做了系统修正:
Function不再默认创建入口块,只有真正定义函数时才建entryIRPrinter对没有基本块的函数输出declare- 自动临时名改成
t0/t1/...,避免 LLVM 对纯数字 SSA 名称的严格顺序约束 - 比较结果按布尔值打印和消费
if/while/and/or生成的块名追加唯一后缀- 修复 float 常量、GEP、Cast、Call 等打印格式
效果
修复后:
simple_add.sy13_sub2.sy29_break.sy36_op_priority2.sy05_arr_defn4.sy
都已经可以通过 verify_ir.sh --run。
4.5 困难五:95_float.sy 的最终运行验证仍受运行库缺失影响
现象
在修完 IR 生成与打印问题后,95_float.sy 已经可以:
- 成功生成 IR
- 通过
llc生成目标文件
但在最终链接阶段仍会失败,原因不是 IR 错误,而是仓库中的 sylib/sylib.c 当前只是空壳,没有提供:
getfloatputfloatgetfarrayputfarrayputchputint
等符号的真实实现。
解决办法
本次提交中没有擅自扩展运行库,而是把问题明确定位为“Lab2 IR 生成正确,但运行时依赖未补齐”。这样可以把 Lab2 编译器部分与后续运行库实现清晰分开。
影响
95_float.sy 当前的状态是:
- IR 生成正确
- LLVM 后端接受
- 最终运行依赖运行库补全
5. 本次实现的主要能力
本次实验结束后,编译器已经具备以下 Lab2 关键能力:
- 全局变量/常量 IR 生成
- 局部变量 IR 生成
int/float常量与表达式生成- 基本算术与比较运算
- 类型转换:
sitofp、fptosi、zext if-elsewhilebreak/continue- 函数定义与函数调用
- 标量参数与数组参数
- 多维数组寻址
- 局部数组零初始化与花括号初始化
- 短路求值
- LLVM 可接受的 IR 文本打印
6. 验证结果
本次已完成的回归包括:
./scripts/verify_ir.sh test/test_case/functional/simple_add.sy /tmp/ir_simple --run
./scripts/verify_ir.sh test/test_case/functional/13_sub2.sy /tmp/ir_sub2 --run
./scripts/verify_ir.sh test/test_case/functional/29_break.sy /tmp/ir_break --run
./scripts/verify_ir.sh test/test_case/functional/36_op_priority2.sy /tmp/ir_op --run
./scripts/verify_ir.sh test/test_case/functional/05_arr_defn4.sy /tmp/ir_arr --run
这些样例均已通过。
另外:
./build/bin/compiler --emit-ir test/test_case/functional/95_float.sy
可以成功生成 IR,且 IR 能通过 llc,说明浮点常量、浮点表达式、浮点比较、类型转换与数组传参路径已经基本打通。
7. 本次实验中的经验总结
本次 Lab2 最核心的经验有三点:
-
编译期常量求值和运行时 IR 生成必须严格分离。
只要两条路径混在一起,全局初始化和常量表达式一定会出错。 -
数组不能按“只是更大的标量”处理。
数组对象、数组形参、数组元素地址、数组退化指针这几个概念必须明确区分。 -
“能打印 IR”不等于“LLVM 能接受 IR”。
最后一定要走一遍llc/clang,否则很多类型和格式问题会被掩盖。
8. 后续可继续完善的方向
虽然本次已经完成了 Lab2 的主体工作,但还可以继续完善:
- 为
sylib补齐实际运行库实现,打通95_float等 I/O 样例的最终运行 - 为全局数组初始化补完整的常量聚合表示,而不是目前以标量初始化为主
- 进一步统一 IR 中布尔类型的内部表示,减少
i1/i32的兼容分支 - 继续批量回归
test/test_case下更多样例,补齐剩余边界情况
9. 结论
本次 Lab2 已经从“完成约 90%,但被全局初始化与数组/短路问题卡住”的状态,推进到“核心 IR 生成链路可用、典型功能样例可运行验证”的状态。阻塞实验验收的主问题已经被定位并解决,代码结构也比原来更清晰,后续继续做运行库、优化与更大规模回归时会更稳。