feat: 数组+索引 [T;N], arr[i] (P1 #6)
- lexer: TOK_LBRACKET, TOK_RBRACKET - type: TYPE_ARRAY + TypeInfo扩展(element_type/array_size) - ast: AST_INDEX_EXPR, AST_ARRAY_ASSIGN_STMT - parser: parse_type_expr()支持[T;N], Pratt加[索引], 数组元素赋值 - sema: 数组类型检查, 索引必须i64, 元素赋值类型匹配 - codegen: type_info_to_llvm(TYPE_ARRAY), GEP+load/store - 新增集成测试: 18_array.l 测试: 136 通过 (41+15+59+21)
This commit is contained in:
@@ -0,0 +1,10 @@
|
||||
fn main() -> i64 {
|
||||
let arr: [i64; 3] = arr;
|
||||
arr[0] = 10;
|
||||
arr[1] = 20;
|
||||
arr[2] = 30;
|
||||
print_i64(arr[0]);
|
||||
print_i64(arr[1]);
|
||||
print_i64(arr[2]);
|
||||
return 0;
|
||||
}
|
||||
+68
-3
@@ -145,7 +145,7 @@ void test_codegen_struct_decl() {
|
||||
init->type.struct_name = "Point";
|
||||
|
||||
AstNode* let_stmt = ast_make_let(&a, "p", TYPE_UNKNOWN, false, false,
|
||||
init, NULL, loc_at(1, 1));
|
||||
init, NULL, 0, NULL, 0, loc_at(1, 1));
|
||||
|
||||
/* return p.x; */
|
||||
AstNode* p_ident = ast_make_ident(&a, "p", loc_at(1, 1));
|
||||
@@ -200,7 +200,7 @@ void test_codegen_struct_field_access() {
|
||||
init->type.struct_name = "Point";
|
||||
|
||||
AstNode* let_stmt = ast_make_let(&a, "p", TYPE_UNKNOWN, false, false,
|
||||
init, NULL, loc_at(1, 1));
|
||||
init, NULL, 0, NULL, 0, loc_at(1, 1));
|
||||
|
||||
/* return p.y; */
|
||||
AstNode* p_ident = ast_make_ident(&a, "p", loc_at(1, 1));
|
||||
@@ -251,7 +251,7 @@ void test_codegen_enum() {
|
||||
cv->type.kind = TYPE_ENUM;
|
||||
|
||||
AstNode* let_stmt = ast_make_let(&a, "c", TYPE_UNKNOWN, false, false,
|
||||
cv, NULL, loc_at(1, 1));
|
||||
cv, NULL, 0, NULL, 0, loc_at(1, 1));
|
||||
|
||||
/* print_i64(c); */
|
||||
AstNode* c_ident = ast_make_ident(&a, "c", loc_at(1, 1));
|
||||
@@ -284,6 +284,70 @@ void test_codegen_enum() {
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
/* === 数组代码生成测试 === */
|
||||
|
||||
void test_codegen_array() {
|
||||
Arena a = arena_create(1);
|
||||
|
||||
/* 构造 AST:
|
||||
fn main() -> i64 {
|
||||
let arr: [i64; 3] = arr;
|
||||
arr[0] = 10;
|
||||
print_i64(arr[0]);
|
||||
return 0;
|
||||
}
|
||||
*/
|
||||
// let arr: [i64; 3] = arr;
|
||||
AstNode* arr_init = ast_make_ident(&a, "arr", loc_at(1, 1));
|
||||
AstNode* let_stmt = ast_make_let(&a, "arr", TYPE_ARRAY, true, false,
|
||||
arr_init, NULL, TYPE_I64, NULL, 3, loc_at(1, 1));
|
||||
// 手动设置 LET_STMT 的类型(绕过 sema)
|
||||
let_stmt->type.kind = TYPE_ARRAY;
|
||||
let_stmt->type.element_type = TYPE_I64;
|
||||
let_stmt->type.array_size = 3;
|
||||
|
||||
// arr[0] = 10;
|
||||
AstNode* arr_assign = ast_make_array_assign(&a, "arr",
|
||||
ast_make_literal_i64(&a, 0, loc_at(1, 1)),
|
||||
ast_make_literal_i64(&a, 10, loc_at(1, 1)), loc_at(1, 1));
|
||||
|
||||
// arr[0] 表达式(print_i64 的参数)
|
||||
AstNode* arr_ident = ast_make_ident(&a, "arr", loc_at(1, 1));
|
||||
arr_ident->type.kind = TYPE_ARRAY;
|
||||
arr_ident->type.element_type = TYPE_I64;
|
||||
arr_ident->type.array_size = 3;
|
||||
AstNode* idx_expr = ast_make_index_expr(&a, arr_ident,
|
||||
ast_make_literal_i64(&a, 0, loc_at(1, 1)), loc_at(1, 1));
|
||||
idx_expr->type.kind = TYPE_I64; // 元素类型
|
||||
|
||||
// print_i64(arr[0]);
|
||||
AstNode* args[] = { idx_expr };
|
||||
AstNode* print_call = ast_make_call(&a, "print_i64", args, 1, loc_at(1, 1));
|
||||
|
||||
// return 0;
|
||||
AstNode* ret = ast_make_return(&a, ast_make_literal_i64(&a, 0, loc_at(1, 1)), loc_at(1, 1));
|
||||
|
||||
AstNode* stmts[] = { let_stmt, arr_assign, print_call, ret };
|
||||
AstNode* body = ast_make_block(&a, stmts, 4, loc_at(1, 1));
|
||||
AstNode* fn = ast_make_function(&a, "main", NULL, 0, TYPE_I64, NULL, body, loc_at(1, 1));
|
||||
AstNode* fns[] = { fn };
|
||||
AstNode* prog = ast_make_program(&a, fns, 1, NULL, 0, NULL, 0, NULL, 0, loc_at(1, 1));
|
||||
|
||||
const char* err = NULL;
|
||||
LLVMContextRef ctx = NULL;
|
||||
LLVMModuleRef mod = codegen_module(prog, &a, "test_array", &err, &ctx);
|
||||
ASSERT(mod != NULL);
|
||||
ASSERT(err == NULL);
|
||||
|
||||
char* verify_err = NULL;
|
||||
int failed = LLVMVerifyModule(mod, LLVMReturnStatusAction, &verify_err);
|
||||
ASSERT(!failed);
|
||||
|
||||
LLVMDisposeModule(mod);
|
||||
LLVMContextDispose(ctx);
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
int main(void) {
|
||||
TEST_RUN(test_codegen_simple_function);
|
||||
TEST_RUN(test_codegen_if_else);
|
||||
@@ -292,5 +356,6 @@ int main(void) {
|
||||
TEST_RUN(test_codegen_struct_decl);
|
||||
TEST_RUN(test_codegen_struct_field_access);
|
||||
TEST_RUN(test_codegen_enum);
|
||||
TEST_RUN(test_codegen_array);
|
||||
return test_summary();
|
||||
}
|
||||
|
||||
@@ -259,6 +259,76 @@ void test_enum_bad_variant() {
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
/* === 数组语义分析测试 === */
|
||||
|
||||
void test_array_ok() {
|
||||
Arena a = arena_create(1);
|
||||
size_t tc; ErrorInfo lex_err = {0};
|
||||
Token* toks = lex(&a,
|
||||
"fn main() { let arr: [i64; 3] = arr; arr[0]; return; }",
|
||||
"test", &tc, &lex_err);
|
||||
ASSERT(toks != NULL);
|
||||
ErrorInfo parse_err = {0};
|
||||
AstNode* ast = parse(&a, toks, tc, "test", &parse_err);
|
||||
ASSERT(ast != NULL);
|
||||
|
||||
ErrorList errors; error_init(&errors);
|
||||
sema_analyze(ast, &errors, &a);
|
||||
ASSERT(errors.count == 0);
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
void test_array_index_type_error() {
|
||||
Arena a = arena_create(1);
|
||||
size_t tc; ErrorInfo lex_err = {0};
|
||||
Token* toks = lex(&a,
|
||||
"fn main() { let arr: [i64; 3] = arr; arr[true]; return; }",
|
||||
"test", &tc, &lex_err);
|
||||
ASSERT(toks != NULL);
|
||||
ErrorInfo parse_err = {0};
|
||||
AstNode* ast = parse(&a, toks, tc, "test", &parse_err);
|
||||
ASSERT(ast != NULL);
|
||||
|
||||
ErrorList errors; error_init(&errors);
|
||||
sema_analyze(ast, &errors, &a);
|
||||
ASSERT(errors.count > 0); // true 不是 i64
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
void test_array_not_indexable() {
|
||||
Arena a = arena_create(1);
|
||||
size_t tc; ErrorInfo lex_err = {0};
|
||||
Token* toks = lex(&a,
|
||||
"fn main() { let x: i64 = 0; x[0]; return; }",
|
||||
"test", &tc, &lex_err);
|
||||
ASSERT(toks != NULL);
|
||||
ErrorInfo parse_err = {0};
|
||||
AstNode* ast = parse(&a, toks, tc, "test", &parse_err);
|
||||
ASSERT(ast != NULL);
|
||||
|
||||
ErrorList errors; error_init(&errors);
|
||||
sema_analyze(ast, &errors, &a);
|
||||
ASSERT(errors.count > 0); // i64 不是数组
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
void test_array_assign_ok() {
|
||||
Arena a = arena_create(1);
|
||||
size_t tc; ErrorInfo lex_err = {0};
|
||||
Token* toks = lex(&a,
|
||||
"fn main() { let arr: [i64; 3] = arr; arr[0] = 42; return; }",
|
||||
"test", &tc, &lex_err);
|
||||
ASSERT(toks != NULL);
|
||||
ErrorInfo parse_err = {0};
|
||||
AstNode* ast = parse(&a, toks, tc, "test", &parse_err);
|
||||
ASSERT(ast != NULL);
|
||||
|
||||
ErrorList errors; error_init(&errors);
|
||||
sema_analyze(ast, &errors, &a);
|
||||
ASSERT(errors.count == 0);
|
||||
arena_destroy(&a);
|
||||
}
|
||||
|
||||
int main(void) {
|
||||
TEST_RUN(test_type_error);
|
||||
TEST_RUN(test_undefined_var);
|
||||
@@ -275,5 +345,9 @@ int main(void) {
|
||||
TEST_RUN(test_type_alias_struct);
|
||||
TEST_RUN(test_enum_ok);
|
||||
TEST_RUN(test_enum_bad_variant);
|
||||
TEST_RUN(test_array_ok);
|
||||
TEST_RUN(test_array_index_type_error);
|
||||
TEST_RUN(test_array_not_indexable);
|
||||
TEST_RUN(test_array_assign_ok);
|
||||
return test_summary();
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user