3b7bab1e1b
5 阶段编译流水线: 词法分析 → 语法分析(Pratt) → 语义分析(类型推断) → LLVM IR → .exe 模块: - lexer: 手写状态机, 40 种 Token, // 和 /* */ 注释 - parser: Pratt 表达式解析(9 级优先级) + 递归下降语句/函数 - ast: 14 种节点类型 + 工厂函数 - sema: 作用域链符号表 + 类型推断 + 类型检查 - codegen: AST → LLVM-C API, print_i64/f64/bool 内建 - driver: 命令行 + 流水线串联 + 错误报告 - util: Arena bump allocator (8MB) 测试: 65 单元测试(词法41+语法15+语义9) + 5 集成测试 全部通过 语言特性: i64/f64/bool/void, let不可变变量, if/else, while, 递归函数
60 lines
1.7 KiB
C
60 lines
1.7 KiB
C
#include "test_utils.h"
|
|
#include "parser.h"
|
|
#include "lexer.h"
|
|
#include "sema.h"
|
|
#include "arena.h"
|
|
|
|
void test_type_error() {
|
|
Arena a = arena_create(1);
|
|
size_t tc; ErrorInfo lex_err = {0};
|
|
Token* toks = lex(&a, "fn main() { let x: i64 = 1; let y: i64 = x + 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);
|
|
arena_destroy(&a);
|
|
}
|
|
|
|
void test_undefined_var() {
|
|
Arena a = arena_create(1);
|
|
size_t tc; ErrorInfo lex_err = {0};
|
|
Token* toks = lex(&a, "fn main() { let x: i64 = y; 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_simple_ok() {
|
|
Arena a = arena_create(1);
|
|
size_t tc; ErrorInfo lex_err = {0};
|
|
Token* toks = lex(&a, "fn main() { let x: i64 = 42; print_i64(x); 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);
|
|
TEST_RUN(test_simple_ok);
|
|
return test_summary();
|
|
}
|