feat(ast2ir): 添加左值标识支持以改善表达式处理
- 在 scc_ast2ir_expr 函数中添加 is_lvalue 参数来区分左值和右值表达式 - 更新二元表达式处理逻辑,特别是赋值操作符的处理 - 改进标识符表达式的处理,根据是否为左值决定返回存储位置还是加载值 - 修复哈希比较函数的实现 - 移除调试相关的注释代码 refactor(parser): 优化语法分析器错误处理和控制流 - 移除不必要的错误恢复辅助注释 - 修改表达式解析的控制流程,将直接返回改为使用 break 语句 - 添加语义分析回调,在解析完成后进行标识符查找和验证 refactor(sema): 增强语义分析阶段的符号表管理 - 改进标识符查找逻辑,增加对非变量标识符的检查 - 扩展声明处理范围,包括变量和参数声明的符号表注册 - 为函数声明添加作用域管理 fix(parser): 修正单元测试中的类型定义 - 将 long long 类型定义改为 int 类型,解决测试兼容性问题 refactor(sccf): 重构文件格式定义和构建器实现 - 重命名符号类型枚举值 OBJECT 为 EXTERN - 重命名段类型枚举值 RELOC 为 RELOCS - 修正结构体字段命名的一致性问题 - 重新设计 SCCF 构建器的数据结构和API - 添加符号表、字符串表和重定位表的构建支持 refactor(target): 重命名Windows PE相关类型定义 - 将 scc_winpe_* 类型重命名为 scc_pe_* 以保持命名一致性 chore: 添加 sccf2target 模块用于格式转换 - 创建新的库模块用于 SCCF 到目标格式的转换 - 实现 PE 格式转换的基本功能 - 添加示例程序演示格式转换过程
This commit is contained in:
@@ -18,7 +18,8 @@ void scc_ast2ir_ctx_init(scc_ast2ir_ctx_t *ctx, const scc_type_abi_t *abi);
|
||||
void scc_ast2ir_translation_unit(scc_ast2ir_ctx_t *ctx,
|
||||
scc_ast_translation_unit_t *tu);
|
||||
void scc_ast2ir_decl(scc_ast2ir_ctx_t *ctx, scc_ast_decl_t *decl);
|
||||
scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr);
|
||||
scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr,
|
||||
cbool is_lvalue);
|
||||
void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt);
|
||||
scc_ir_type_ref_t scc_ast2ir_type(scc_ast2ir_ctx_t *ctx,
|
||||
scc_ast_type_t *ast_type);
|
||||
|
||||
@@ -92,7 +92,8 @@ scc_ir_type_ref_t scc_ast2ir_type(scc_ast2ir_ctx_t *ctx,
|
||||
* @param expr
|
||||
* @return scc_ir_node_ref_t
|
||||
*/
|
||||
scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr,
|
||||
cbool is_lvalue) {
|
||||
if (ctx == null || expr == null) {
|
||||
LOG_ERROR("args is null");
|
||||
return 0;
|
||||
@@ -101,8 +102,27 @@ scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
switch (expr->base.type) {
|
||||
case SCC_AST_EXPR_BINARY: {
|
||||
scc_ir_node_ref_t lhs, rhs;
|
||||
lhs = scc_ast2ir_expr(ctx, expr->binary.lhs);
|
||||
rhs = scc_ast2ir_expr(ctx, expr->binary.rhs);
|
||||
cbool is_assign = false;
|
||||
switch (expr->binary.op) {
|
||||
case SCC_AST_OP_ASSIGN: // =
|
||||
case SCC_AST_OP_ASSIGN_ADD: // +=
|
||||
case SCC_AST_OP_ASSIGN_SUB: // -=
|
||||
case SCC_AST_OP_ASSIGN_MUL: // *=
|
||||
case SCC_AST_OP_ASSIGN_DIV: // /=
|
||||
case SCC_AST_OP_ASSIGN_MOD: // %=
|
||||
case SCC_AST_OP_ASSIGN_AND: // &=
|
||||
case SCC_AST_OP_ASSIGN_XOR: // ^=
|
||||
case SCC_AST_OP_ASSIGN_OR: // |=
|
||||
case SCC_AST_OP_ASSIGN_LSHIFT: // <<=
|
||||
case SCC_AST_OP_ASSIGN_RSHIFT: // >>=
|
||||
is_assign = true;
|
||||
break;
|
||||
default:
|
||||
is_assign = false;
|
||||
break;
|
||||
}
|
||||
lhs = scc_ast2ir_expr(ctx, expr->binary.lhs, is_assign);
|
||||
rhs = scc_ast2ir_expr(ctx, expr->binary.rhs, false);
|
||||
|
||||
// 映射操作符
|
||||
scc_ir_op_type_t op;
|
||||
@@ -124,17 +144,6 @@ scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
case SCC_AST_OP_LESS_EQUAL: op = SCC_IR_OP_LE; break;
|
||||
case SCC_AST_OP_GREATER: op = SCC_IR_OP_GT; break;
|
||||
case SCC_AST_OP_GREATER_EQUAL: op = SCC_IR_OP_GE; break;
|
||||
// SCC_AST_OP_ASSIGN, // =
|
||||
// SCC_AST_OP_ASSIGN_ADD, // +=
|
||||
// SCC_AST_OP_ASSIGN_SUB, // -=
|
||||
// SCC_AST_OP_ASSIGN_MUL, // *=
|
||||
// SCC_AST_OP_ASSIGN_DIV, // /=
|
||||
// SCC_AST_OP_ASSIGN_MOD, // %=
|
||||
// SCC_AST_OP_ASSIGN_AND, // &=
|
||||
// SCC_AST_OP_ASSIGN_XOR, // ^=
|
||||
// SCC_AST_OP_ASSIGN_OR, // |=
|
||||
// SCC_AST_OP_ASSIGN_LSHIFT, // <<=
|
||||
// SCC_AST_OP_ASSIGN_RSHIFT, // >>=
|
||||
case SCC_AST_OP_ASSIGN: {
|
||||
return scc_ir_builder_store(&ctx->builder, lhs, rhs);
|
||||
}
|
||||
@@ -194,7 +203,8 @@ scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
}
|
||||
|
||||
case SCC_AST_EXPR_UNARY: {
|
||||
scc_ir_node_ref_t operand = scc_ast2ir_expr(ctx, expr->unary.operand);
|
||||
scc_ir_node_ref_t operand =
|
||||
scc_ast2ir_expr(ctx, expr->unary.operand, is_lvalue);
|
||||
|
||||
// 映射一元操作符
|
||||
switch (expr->unary.op) {
|
||||
@@ -235,7 +245,7 @@ scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
scc_vec_foreach(expr->call.args, i) {
|
||||
scc_ast_expr_t *arg_expr = scc_vec_at(expr->call.args, i);
|
||||
scc_ir_node_ref_t arg_node;
|
||||
arg_node = scc_ast2ir_expr(ctx, arg_expr);
|
||||
arg_node = scc_ast2ir_expr(ctx, arg_expr, false);
|
||||
scc_vec_push(args, arg_node);
|
||||
}
|
||||
}
|
||||
@@ -273,10 +283,17 @@ scc_ir_node_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx, scc_ast_expr_t *expr) {
|
||||
// SCC_AST_EXPR_STRING_LITERAL, // 字符串字面量
|
||||
|
||||
case SCC_AST_EXPR_IDENTIFIER: {
|
||||
if (expr->identifier._target == null) {
|
||||
LOG_ERROR("unknown identifier");
|
||||
}
|
||||
// FIXME hack hashtable
|
||||
scc_ir_node_ref_t in = (scc_ir_node_ref_t)(usize)scc_hashtable_get(
|
||||
&ctx->node2ir, expr->identifier._target);
|
||||
return scc_ir_builder_load(&ctx->builder, in);
|
||||
if (is_lvalue) {
|
||||
return in;
|
||||
} else {
|
||||
return scc_ir_builder_load(&ctx->builder, in);
|
||||
}
|
||||
}
|
||||
|
||||
default:
|
||||
@@ -315,7 +332,7 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt) {
|
||||
}
|
||||
|
||||
case SCC_AST_STMT_EXPR: {
|
||||
scc_ast2ir_expr(ctx, stmt->expr.expr);
|
||||
scc_ast2ir_expr(ctx, stmt->expr.expr, false);
|
||||
break;
|
||||
}
|
||||
|
||||
@@ -336,7 +353,8 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt) {
|
||||
scc_ir_bblock_ref_t merge_block =
|
||||
scc_ir_builder_bblock(&ctx->builder, "if_merge");
|
||||
|
||||
scc_ir_node_ref_t cond_node = scc_ast2ir_expr(ctx, stmt->if_stmt.cond);
|
||||
scc_ir_node_ref_t cond_node =
|
||||
scc_ast2ir_expr(ctx, stmt->if_stmt.cond, false);
|
||||
scc_ir_builder_branch(&ctx->builder, cond_node, true_block,
|
||||
false_block);
|
||||
|
||||
@@ -368,7 +386,7 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt) {
|
||||
|
||||
scc_ir_builder_set_current_bblock(&ctx->builder, cond_block);
|
||||
scc_ir_node_ref_t cond_node =
|
||||
scc_ast2ir_expr(ctx, stmt->while_stmt.cond);
|
||||
scc_ast2ir_expr(ctx, stmt->while_stmt.cond, false);
|
||||
scc_ir_builder_branch(&ctx->builder, cond_node, body_block, exit_block);
|
||||
|
||||
scc_ir_builder_set_current_bblock(&ctx->builder, body_block);
|
||||
@@ -395,7 +413,7 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt) {
|
||||
|
||||
scc_ir_builder_set_current_bblock(&ctx->builder, cond_block);
|
||||
scc_ir_node_ref_t cond_node =
|
||||
scc_ast2ir_expr(ctx, stmt->while_stmt.cond);
|
||||
scc_ast2ir_expr(ctx, stmt->while_stmt.cond, false);
|
||||
scc_ir_builder_branch(&ctx->builder, cond_node, body_block, exit_block);
|
||||
|
||||
scc_ir_builder_set_current_bblock(&ctx->builder, exit_block);
|
||||
@@ -421,7 +439,7 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, scc_ast_stmt_t *stmt) {
|
||||
case SCC_AST_STMT_RETURN: {
|
||||
if (stmt->return_stmt.expr) {
|
||||
scc_ir_node_ref_t ret_val_node =
|
||||
scc_ast2ir_expr(ctx, stmt->return_stmt.expr);
|
||||
scc_ast2ir_expr(ctx, stmt->return_stmt.expr, false);
|
||||
scc_ir_builder_ret(&ctx->builder, ret_val_node);
|
||||
} else {
|
||||
scc_ir_builder_ret_void(&ctx->builder);
|
||||
@@ -457,11 +475,14 @@ void scc_ast2ir_decl(scc_ast2ir_ctx_t *ctx, scc_ast_decl_t *decl) {
|
||||
scc_ir_node_ref_t alloc_val_node =
|
||||
scc_ir_builder_alloca(&ctx->builder, ir_type, decl->name);
|
||||
|
||||
scc_hashtable_set(&ctx->node2ir, decl, (void *)(usize)alloc_val_node);
|
||||
|
||||
// 如果有初始化表达式
|
||||
if (!decl->var.init) {
|
||||
break;
|
||||
}
|
||||
scc_ir_node_ref_t init_val_node = scc_ast2ir_expr(ctx, decl->var.init);
|
||||
scc_ir_node_ref_t init_val_node =
|
||||
scc_ast2ir_expr(ctx, decl->var.init, false);
|
||||
scc_ir_builder_store(&ctx->builder, alloc_val_node, init_val_node);
|
||||
break;
|
||||
}
|
||||
@@ -518,7 +539,7 @@ void scc_ast2ir_translation_unit(scc_ast2ir_ctx_t *ctx,
|
||||
|
||||
static u32 scc_hash_node(const void *key) { return (u32)(usize)key; }
|
||||
static int scc_cmp_node(const void *key1, const void *key2) {
|
||||
return key1 == key2;
|
||||
return (u32)(usize)key1 - (u32)(usize)key2;
|
||||
}
|
||||
|
||||
void scc_ast2ir_ctx_init(scc_ast2ir_ctx_t *ctx, const scc_type_abi_t *abi) {
|
||||
|
||||
Reference in New Issue
Block a user