refactor(ir): 改一下sema相关的适配

This commit is contained in:
jing
2026-03-17 19:09:21 +08:00
parent f19d23c656
commit 1b283856b3

View File

@@ -11,57 +11,74 @@ namespace {
void CheckExpr(SysYParser::ExpContext& exp, const SymbolTable& table, void CheckExpr(SysYParser::ExpContext& exp, const SymbolTable& table,
SemanticContext& sema); SemanticContext& sema);
void CheckPrimary(SysYParser::PrimaryContext& primary, std::string GetLValueName(SysYParser::LValueContext& lvalue) {
const SymbolTable& table, SemanticContext& sema) { if (!lvalue.ID()) {
if (primary.Number()) { throw std::runtime_error(FormatError("sema", "非法左值"));
return;
} }
return lvalue.ID()->getText();
}
if (primary.Ident()) { void CheckVar(SysYParser::VarContext& var, const SymbolTable& table,
const std::string name = primary.Ident()->getText(); SemanticContext& sema) {
if (!var.ID()) {
throw std::runtime_error(FormatError("sema", "非法变量引用"));
}
const std::string name = var.ID()->getText();
auto* decl = table.Lookup(name); auto* decl = table.Lookup(name);
if (!decl) { if (!decl) {
throw std::runtime_error(FormatError("sema", "使用了未定义的变量: " + name)); throw std::runtime_error(FormatError("sema", "使用了未定义的变量: " + name));
} }
sema.BindVarUse(&primary, decl); sema.BindVarUse(&var, decl);
return;
}
if (primary.exp()) {
CheckExpr(*primary.exp(), table, sema);
return;
}
throw std::runtime_error(FormatError("sema", "暂不支持的表达式形式"));
} }
void CheckExpr(SysYParser::ExpContext& exp, const SymbolTable& table, void CheckExpr(SysYParser::ExpContext& exp, const SymbolTable& table,
SemanticContext& sema) { SemanticContext& sema) {
if (!exp.addExp()) { if (auto* paren = dynamic_cast<SysYParser::ParenExpContext*>(&exp)) {
throw std::runtime_error(FormatError("sema", "非法表达式")); CheckExpr(*paren->exp(), table, sema);
return;
} }
const auto& terms = exp.addExp()->primary(); if (auto* var = dynamic_cast<SysYParser::VarExpContext*>(&exp)) {
for (auto* term : terms) { if (!var->var()) {
CheckPrimary(*term, table, sema); throw std::runtime_error(FormatError("sema", "非法变量表达式"));
} }
CheckVar(*var->var(), table, sema);
return;
}
if (dynamic_cast<SysYParser::NumberExpContext*>(&exp)) {
return;
}
if (auto* binary = dynamic_cast<SysYParser::AdditiveExpContext*>(&exp)) {
CheckExpr(*binary->exp(0), table, sema);
CheckExpr(*binary->exp(1), table, sema);
return;
}
throw std::runtime_error(FormatError("sema", "暂不支持的表达式形式"));
}
SysYParser::FuncDefContext* FindMainFunc(SysYParser::CompUnitContext& comp_unit) {
auto* func = comp_unit.funcDef();
if (func && func->ID() && func->ID()->getText() == "main") {
return func;
}
return nullptr;
} }
} // namespace } // namespace
SemanticContext RunSema(SysYParser::CompUnitContext& comp_unit) { SemanticContext RunSema(SysYParser::CompUnitContext& comp_unit) {
auto* func = comp_unit.funcDef(); auto* func = FindMainFunc(comp_unit);
if (!func || !func->block()) { if (!func || !func->blockStmt()) {
throw std::runtime_error(FormatError("sema", "缺少 main 函数定义")); throw std::runtime_error(FormatError("sema", "缺少 main 函数定义"));
} }
if (!func->Ident() || func->Ident()->getText() != "main") { if (!func->funcType() || !func->funcType()->INT()) {
throw std::runtime_error(FormatError("sema", "入口函数必须命名为 main")); throw std::runtime_error(FormatError("sema", "当前仅支持 int main"));
} }
SymbolTable table; SymbolTable table;
SemanticContext sema; SemanticContext sema;
bool seen_return = false; bool seen_return = false;
const auto& items = func->block()->blockItem(); const auto& items = func->blockStmt()->blockItem();
if (items.empty()) { if (items.empty()) {
throw std::runtime_error( throw std::runtime_error(
FormatError("sema", "main 函数不能为空,且必须以 return 结束")); FormatError("sema", "main 函数不能为空,且必须以 return 结束"));
@@ -76,19 +93,33 @@ SemanticContext RunSema(SysYParser::CompUnitContext& comp_unit) {
throw std::runtime_error( throw std::runtime_error(
FormatError("sema", "return 必须是 main 函数中的最后一条语句")); FormatError("sema", "return 必须是 main 函数中的最后一条语句"));
} }
if (auto* decl = item->decl() ? item->decl()->varDecl() : nullptr) { if (auto* decl = item->decl()) {
const std::string name = decl->Ident()->getText(); if (!decl->btype() || !decl->btype()->INT()) {
throw std::runtime_error(FormatError("sema", "当前仅支持局部 int 变量声明"));
}
auto* var_def = decl->varDef();
if (!var_def || !var_def->lValue()) {
throw std::runtime_error(FormatError("sema", "非法变量声明"));
}
const std::string name = GetLValueName(*var_def->lValue());
if (table.Contains(name)) { if (table.Contains(name)) {
throw std::runtime_error(FormatError("sema", "重复定义变量: " + name)); throw std::runtime_error(FormatError("sema", "重复定义变量: " + name));
} }
if (decl->exp()) { if (auto* init = var_def->initValue()) {
CheckExpr(*decl->exp(), table, sema); if (!init->exp()) {
throw std::runtime_error(
FormatError("sema", "当前不支持聚合初始化"));
} }
table.Add(name, decl); CheckExpr(*init->exp(), table, sema);
}
table.Add(name, var_def);
continue; continue;
} }
if (auto* stmt = item->stmt(); stmt && stmt->returnStmt()) { if (auto* stmt = item->stmt(); stmt && stmt->returnStmt()) {
auto* ret = stmt->returnStmt(); auto* ret = stmt->returnStmt();
if (!ret->exp()) {
throw std::runtime_error(FormatError("sema", "return 缺少表达式"));
}
CheckExpr(*ret->exp(), table, sema); CheckExpr(*ret->exp(), table, sema);
seen_return = true; seen_return = true;
if (i + 1 != items.size()) { if (i + 1 != items.size()) {