- 引入了 WhileStmt 结构来表示 while 循环语句。 - 在解析器中实现了对 while 语句的解析逻辑。 - 在分析器中为 while 语句添加了语义分析。 - 重构了现有的解析器方法,以利用 StateProtector 进行状态管理。 - 更新了对各种表达式和语句的错误处理。 - 移除了未使用的终止符管理方法,并简化了表达式解析。 - 将 FigLSPServer.cpp 重命名为 LSPServer.cpp,并调整了构建配置。 - 增强了重复声明和类型错误的错误报告。 - 在多个文件中改进了代码格式和一致性。
212 lines
5.8 KiB
C++
212 lines
5.8 KiB
C++
/*!
|
||
@file src/Compiler/StmtCompiler.cpp
|
||
@brief 编译器实现(语句部分)
|
||
@author PuqiAR (im@puqiar.top)
|
||
@date 2026-02-19
|
||
*/
|
||
|
||
#include <Compiler/Compiler.hpp>
|
||
|
||
namespace Fig
|
||
{
|
||
Result<void, Error> Compiler::compileVarDecl(VarDecl *varDecl)
|
||
{
|
||
if (current->freeReg > MAX_LOCALS)
|
||
{
|
||
return std::unexpected(Error(ErrorType::TooManyLocals,
|
||
std::format("local limit exceeded: {}", MAX_LOCALS),
|
||
"try split function or use arrays/structs...",
|
||
makeSourceLocation(varDecl)));
|
||
}
|
||
|
||
std::uint8_t varReg;
|
||
if (varDecl->initExpr)
|
||
{
|
||
auto result = compileExpr(varDecl->initExpr);
|
||
if (!result)
|
||
{
|
||
return std::unexpected(result.error());
|
||
}
|
||
std::uint8_t resultReg = *result;
|
||
varReg = DeclareLocal(varDecl->localId, resultReg); // 复用临时计算结果寄存器
|
||
}
|
||
else
|
||
{
|
||
varReg = DeclareLocal(varDecl->localId);
|
||
}
|
||
|
||
return Result<void, Error>();
|
||
}
|
||
|
||
Result<void, Error> Compiler::compileBlockStmt(BlockStmt *blockStmt)
|
||
{
|
||
for (Stmt *stmt : blockStmt->nodes)
|
||
{
|
||
auto result = compileStmt(stmt);
|
||
if (!result)
|
||
{
|
||
return result;
|
||
}
|
||
}
|
||
return {};
|
||
}
|
||
|
||
Result<void, Error> Compiler::compileIfStmt(IfStmt *stmt)
|
||
{
|
||
/*
|
||
if cond1
|
||
{
|
||
}
|
||
else if cond2 #1
|
||
{
|
||
}
|
||
else if cond3 #2
|
||
{
|
||
}
|
||
else #3
|
||
{
|
||
}
|
||
quit #4
|
||
|
||
Bytecode:
|
||
JmpIfFalse cond1 #1
|
||
; consequent内容
|
||
; ...
|
||
; if条件为true, 跳过所有 else/elseif
|
||
|
||
Jmp #4
|
||
|
||
; #1
|
||
JmpIfFalse cond2 #2
|
||
; consequent
|
||
|
||
Jmp #4
|
||
|
||
; #2
|
||
JmpIfFalse cond3 #3
|
||
; consequent
|
||
|
||
Jmp #4
|
||
|
||
; #3
|
||
; 没有一次执行分支
|
||
; else部分
|
||
; ...
|
||
|
||
#4
|
||
|
||
*/
|
||
std::vector<int> exitJumps; // 所有分支都要跳到最后,收集所有jump最后回填
|
||
const auto &condResult = compileExpr(stmt->cond);
|
||
if (!condResult)
|
||
{
|
||
return std::unexpected(condResult.error());
|
||
}
|
||
std::uint8_t condReg = *condResult;
|
||
int jumpToNext = EmitJump(OpCode::JmpIfFalse, condReg);
|
||
FreeReg(condReg);
|
||
|
||
const auto &blockResult = compileStmt(stmt->consequent);
|
||
if (!blockResult)
|
||
{
|
||
return blockResult;
|
||
}
|
||
|
||
exitJumps.push_back(EmitJump(OpCode::Jmp)); // 执行完if直接跳到出口
|
||
PatchJump(jumpToNext); // 回填,跳到下一个else/elseif
|
||
|
||
for (auto *elif : stmt->elifs)
|
||
{
|
||
const auto &elifCondResult = compileExpr(elif->cond);
|
||
if (!elifCondResult)
|
||
return std::unexpected(elifCondResult.error());
|
||
std::uint8_t elifCondReg = *elifCondResult;
|
||
|
||
jumpToNext = EmitJump(OpCode::JmpIfFalse, elifCondReg);
|
||
FreeReg(elifCondReg);
|
||
|
||
const auto &blockResult = compileStmt(elif->consequent);
|
||
if (!blockResult)
|
||
{
|
||
return blockResult;
|
||
}
|
||
exitJumps.push_back(EmitJump(OpCode::Jmp)); // 执行完else if,跳到出口
|
||
PatchJump(jumpToNext); // 跳到下一个分支
|
||
}
|
||
|
||
if (stmt->alternate)
|
||
{
|
||
auto result = compileStmt(stmt->alternate);
|
||
if (!result)
|
||
{
|
||
return result;
|
||
}
|
||
}
|
||
for (int exitIndex : exitJumps)
|
||
{
|
||
PatchJump(exitIndex); // 回填所有跳转出口的指令
|
||
}
|
||
return {};
|
||
}
|
||
|
||
Result<void, Error> Compiler::compileWhileStmt(WhileStmt *stmt)
|
||
{
|
||
int beginIns = current->proto->code.size() - 1;
|
||
|
||
auto condRegResult = compileExpr(stmt->cond);
|
||
if (!condRegResult)
|
||
{
|
||
return std::unexpected(condRegResult.error());
|
||
}
|
||
|
||
std::uint8_t condReg = *condRegResult;
|
||
|
||
int exitJump = EmitJump(OpCode::JmpIfFalse, condReg);
|
||
auto bodyResult = compileBlockStmt(stmt->body);
|
||
|
||
if (!bodyResult)
|
||
{
|
||
return bodyResult;
|
||
}
|
||
Emit(Op::iAsBx(
|
||
OpCode::Jmp, 0, beginIns - current->proto->code.size())); // 回到开头对condition求值
|
||
PatchJump(exitJump);
|
||
return {};
|
||
}
|
||
|
||
Result<void, Error> Compiler::compileStmt(Stmt *stmt) // 编译语句
|
||
{
|
||
switch (stmt->type)
|
||
{
|
||
case AstType::ExprStmt: {
|
||
ExprStmt *exprStmt = static_cast<ExprStmt *>(stmt);
|
||
Expr *expr = exprStmt->expr;
|
||
auto result = compileExpr(expr);
|
||
if (!result)
|
||
{
|
||
return std::unexpected(result.error());
|
||
}
|
||
FreeReg(*result);
|
||
break;
|
||
}
|
||
|
||
case AstType::VarDecl: {
|
||
return compileVarDecl(static_cast<VarDecl *>(stmt));
|
||
}
|
||
|
||
case AstType::BlockStmt: {
|
||
return compileBlockStmt(static_cast<BlockStmt *>(stmt));
|
||
}
|
||
|
||
case AstType::IfStmt: {
|
||
return compileIfStmt(static_cast<IfStmt *>(stmt));
|
||
}
|
||
|
||
case AstType::WhileStmt: {
|
||
return compileWhileStmt(static_cast<WhileStmt *>(stmt));
|
||
}
|
||
}
|
||
|
||
return Result<void, Error>();
|
||
}
|
||
}; // namespace Fig
|