feat(abi): 添加VA_LIST类型支持并完善ABI实现
- 在scc_abi_base_type_kind枚举中添加SCC_ABI_TYPE_VA_LIST类型 - 为Windows x64平台实现VA_LIST类型的ABI规则,遵循MSVC调用约定 - 在AST到ABI类型转换中处理SCC_AST_BUILTIN_TYPE_VA_LIST和BOOL类型 refactor(ast2ir): 实现循环控制流和跳转语句转换 - 添加break和continue缓存表用于语句跳转目标管理 - 实现while、do-while和for循环的正确控制流结构 - 添加对break、continue、goto和label语句的IR转换支持 - 修复复合表达式和声明类型的处理逻辑 refactor(parser): 重构语义分析接口和循环语句解析 - 将语义分析上下文从回调结构改为指针传递 - 为switch、while、do-while和for语句添加BEGIN/END语义标记 - 实现匿名结构体/联合体/枚举的符号命名处理 - 优化结构体/联合体/枚举声明的类型解析逻辑 feat(parser): 实现跳转语句语义分析和符号绑定 - 添加break_stack和continue_stack用于跟踪循环层级 - 实现break和continue语句的目标语句绑定检查 - 支持goto和label语句的标签符号查找和绑定 - 添加对跳转语句位置的有效性验证 Co-authored-by: Copilot <copilot@github.com>
This commit is contained in:
@@ -7,8 +7,10 @@
|
||||
|
||||
typedef struct {
|
||||
scc_hir_builder_t builder;
|
||||
scc_hashtable_t ast2ir_cache; ///< ast node to ir ref cache
|
||||
scc_hashtable_t symtab; ///< symbol to ir_ref
|
||||
scc_hashtable_t ast2ir_cache; ///< ast node to ir ref cache
|
||||
scc_hashtable_t break_cache; ///< break cache
|
||||
scc_hashtable_t continue_cache; ///< continue cache
|
||||
scc_hashtable_t symtab; ///< symbol to ir_ref
|
||||
// scc_strpool_t strpool; ///< string pool
|
||||
const scc_abi_type_calc_t *abi;
|
||||
cbool hint_using_value; // 转换时尽可能使用value而不是alloc
|
||||
|
||||
@@ -626,8 +626,7 @@ scc_hir_value_ref_t scc_ast2ir_expr(scc_ast2ir_ctx_t *ctx,
|
||||
// &ctx->builder, scc_hir_builder_type_u64(&ctx->builder), val);
|
||||
}
|
||||
case SCC_AST_EXPR_COMPOUND: {
|
||||
break;
|
||||
}
|
||||
} break;
|
||||
case SCC_AST_EXPR_LVALUE:
|
||||
break;
|
||||
// SCC_AST_EXPR_BUILTIN,// 内置表达式 ... directive map to ir builtin
|
||||
@@ -793,6 +792,9 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
scc_hir_value_ref_t exit_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "while_exit");
|
||||
|
||||
scc_hashtable_set(&ctx->break_cache, stmt, (void *)exit_block);
|
||||
scc_hashtable_set(&ctx->continue_cache, stmt, (void *)cond_block);
|
||||
|
||||
scc_hir_builder_jump(&ctx->builder, cond_block);
|
||||
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, cond_block);
|
||||
@@ -816,6 +818,9 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
scc_hir_value_ref_t exit_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "do_while_exit");
|
||||
|
||||
scc_hashtable_set(&ctx->break_cache, stmt, (void *)exit_block);
|
||||
scc_hashtable_set(&ctx->continue_cache, stmt, (void *)cond_block);
|
||||
|
||||
scc_hir_builder_jump(&ctx->builder, body_block);
|
||||
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, body_block);
|
||||
@@ -833,11 +838,16 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
}
|
||||
case SCC_AST_STMT_FOR: {
|
||||
scc_hir_value_ref_t cond_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_while_cond");
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_cond");
|
||||
scc_hir_value_ref_t body_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_while_body");
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_body");
|
||||
scc_hir_value_ref_t incr_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_incr");
|
||||
scc_hir_value_ref_t exit_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_while_exit");
|
||||
scc_hir_builder_bblock(&ctx->builder, "for_exit");
|
||||
|
||||
scc_hashtable_set(&ctx->break_cache, stmt, (void *)exit_block);
|
||||
scc_hashtable_set(&ctx->continue_cache, stmt, (void *)incr_block);
|
||||
|
||||
if (stmt->for_stmt.init) {
|
||||
if (SCC_AST_IS_A(scc_ast_decl_t, stmt->for_stmt.init)) {
|
||||
@@ -867,10 +877,13 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, body_block);
|
||||
scc_ast2ir_stmt(ctx, stmt->for_stmt.body);
|
||||
scc_hir_builder_jump(&ctx->builder, incr_block);
|
||||
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, incr_block);
|
||||
if (stmt->for_stmt.incr) {
|
||||
scc_ast2ir_expr(ctx, stmt->for_stmt.incr, false);
|
||||
}
|
||||
scc_hir_builder_jump(&ctx->builder, cond_block);
|
||||
scc_hir_builder_jump(&ctx->builder, exit_block);
|
||||
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, exit_block);
|
||||
break;
|
||||
@@ -878,8 +891,18 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
// SCC_AST_STMT_SWITCH, // switch 语句
|
||||
// SCC_AST_STMT_CASE, // case 语句
|
||||
// SCC_AST_STMT_DEFAULT, // default 语句
|
||||
// SCC_AST_STMT_BREAK, // break 语句
|
||||
// SCC_AST_STMT_CONTINUE, // continue 语句
|
||||
case SCC_AST_STMT_BREAK: {
|
||||
scc_hir_bblock_ref_t target =
|
||||
(usize)scc_hashtable_get(&ctx->break_cache, stmt->jump._target);
|
||||
Assert(target != SCC_HIR_REF_nullptr);
|
||||
scc_hir_builder_jump(&ctx->builder, target);
|
||||
} break;
|
||||
case SCC_AST_STMT_CONTINUE: {
|
||||
scc_hir_bblock_ref_t target =
|
||||
(usize)scc_hashtable_get(&ctx->continue_cache, stmt->jump._target);
|
||||
Assert(target != SCC_HIR_REF_nullptr);
|
||||
scc_hir_builder_jump(&ctx->builder, target);
|
||||
} break;
|
||||
case SCC_AST_STMT_RETURN: {
|
||||
if (stmt->return_stmt.expr) {
|
||||
scc_hir_value_ref_t ret_val_node =
|
||||
@@ -890,8 +913,19 @@ void scc_ast2ir_stmt(scc_ast2ir_ctx_t *ctx, const scc_ast_stmt_t *stmt) {
|
||||
}
|
||||
break;
|
||||
}
|
||||
// SCC_AST_STMT_GOTO, // goto 语句
|
||||
// SCC_AST_STMT_LABEL, // 标签语句
|
||||
case SCC_AST_STMT_GOTO: {
|
||||
scc_hir_bblock_ref_t target = (usize)scc_hashtable_get(
|
||||
&ctx->ast2ir_cache, (void *)stmt->goto_stmt._target);
|
||||
Assert(target != SCC_HIR_REF_nullptr);
|
||||
scc_hir_builder_jump(&ctx->builder, target);
|
||||
} break;
|
||||
case SCC_AST_STMT_LABEL: {
|
||||
scc_hir_value_ref_t label_block =
|
||||
scc_hir_builder_bblock(&ctx->builder, stmt->label_stmt.label);
|
||||
scc_hir_builder_set_current_bblock(&ctx->builder, label_block);
|
||||
scc_hashtable_set(&ctx->ast2ir_cache, stmt, (void *)label_block);
|
||||
scc_ast2ir_stmt(ctx, stmt->label_stmt.stmt);
|
||||
} break;
|
||||
default:
|
||||
LOG_FATAL("Unsupported statement type: %d", stmt->base.type);
|
||||
break;
|
||||
@@ -1039,7 +1073,7 @@ void scc_ast2ir_decl(scc_ast2ir_ctx_t *ctx, const scc_ast_decl_t *decl,
|
||||
break;
|
||||
}
|
||||
case SCC_AST_DECL_STRUCT:
|
||||
case SCC_AST_DECL_UNION:
|
||||
case SCC_AST_DECL_UNION: {
|
||||
scc_ast_canon_type_t canon_type = {
|
||||
.record.decl = decl,
|
||||
.record.name = decl->name,
|
||||
@@ -1052,8 +1086,8 @@ void scc_ast2ir_decl(scc_ast2ir_ctx_t *ctx, const scc_ast_decl_t *decl,
|
||||
scc_hir_type_ref_t type_ref = scc_ast2ir_type(ctx, &type);
|
||||
// scc_hir_builder_global_alloca(&ctx->builder, type_ref,
|
||||
// SCC_HIR_REF_nullptr);
|
||||
break;
|
||||
case SCC_AST_DECL_ENUM:
|
||||
} break;
|
||||
case SCC_AST_DECL_ENUM: {
|
||||
scc_ap_t val;
|
||||
int idx = 0;
|
||||
scc_ap_set_int(&val, idx);
|
||||
@@ -1070,7 +1104,7 @@ void scc_ast2ir_decl(scc_ast2ir_ctx_t *ctx, const scc_ast_decl_t *decl,
|
||||
(void *)(usize)item_val_ref);
|
||||
scc_ap_set_int(&val, ++idx);
|
||||
}
|
||||
break;
|
||||
} break;
|
||||
case SCC_AST_DECL_TYPEDEF:
|
||||
break;
|
||||
default:
|
||||
@@ -1095,6 +1129,8 @@ void scc_ast2ir_ctx_init(scc_ast2ir_ctx_t *ctx, const scc_abi_type_calc_t *abi,
|
||||
ctx->abi = abi;
|
||||
scc_hir_builder_init(&ctx->builder, cprog);
|
||||
scc_hashtable_usize_init(&ctx->ast2ir_cache);
|
||||
scc_hashtable_usize_init(&ctx->break_cache);
|
||||
scc_hashtable_usize_init(&ctx->continue_cache);
|
||||
scc_hashtable_cstr_init(&ctx->symtab);
|
||||
ctx->hint_using_value = false;
|
||||
}
|
||||
@@ -1102,5 +1138,7 @@ void scc_ast2ir_ctx_init(scc_ast2ir_ctx_t *ctx, const scc_abi_type_calc_t *abi,
|
||||
void scc_ast2ir_ctx_drop(scc_ast2ir_ctx_t *ctx) {
|
||||
scc_hir_builder_drop(&ctx->builder);
|
||||
scc_hashtable_drop(&ctx->ast2ir_cache);
|
||||
scc_hashtable_drop(&ctx->break_cache);
|
||||
scc_hashtable_drop(&ctx->continue_cache);
|
||||
scc_hashtable_drop(&ctx->symtab);
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user