diff options
author | Kimplul <kimi.h.kuparinen@gmail.com> | 2025-03-23 22:29:11 +0200 |
---|---|---|
committer | Kimplul <kimi.h.kuparinen@gmail.com> | 2025-03-23 22:29:11 +0200 |
commit | c87f5a8871edf6880b894a00b180c554ffd46d0a (patch) | |
tree | 8f4ac966d1ca37884bf55078b8318d0ba198af9e /src | |
parent | 350f6c40fa18c35bde9489225175c82de44ba709 (diff) | |
download | fwd-c87f5a8871edf6880b894a00b180c554ffd46d0a.tar.gz fwd-c87f5a8871edf6880b894a00b180c554ffd46d0a.zip |
start sketching out type system
Diffstat (limited to 'src')
-rw-r--r-- | src/analyze.c | 31 | ||||
-rw-r--r-- | src/ast.c | 10 | ||||
-rw-r--r-- | src/lexer.l | 4 | ||||
-rw-r--r-- | src/parser.y | 88 | ||||
-rw-r--r-- | src/scope.c | 139 |
5 files changed, 217 insertions, 55 deletions
diff --git a/src/analyze.c b/src/analyze.c index c33a901..0c06096 100644 --- a/src/analyze.c +++ b/src/analyze.c @@ -189,7 +189,7 @@ static int analyze_var(struct state *state, struct scope *scope, return -1; node->t = var_type(node); - return scope_add_var(scope, node); + return scope_add_symbol(scope, node); } static int analyze_let(struct state *state, struct scope *scope, @@ -478,7 +478,7 @@ static int analyze_err_branch(struct state *state, struct scope *scope, struct a struct ast *var = gen_var(strdup(err_branch_id(node)), NULL, node->loc); struct type *err_type = tgen_err(node->loc); var->t = err_type; - scope_add_var(branch_scope, var); + scope_add_symbol(branch_scope, var); return analyze(state, branch_scope, err_branch_body(node)); } @@ -607,9 +607,30 @@ static int analyze_type_list(struct state *state, struct scope *scope, int analyze_root(struct scope *scope, struct ast *root) { foreach_node(node, root) { - assert(node->k == AST_PROC_DEF); - if (scope_add_proc(scope, node)) - return -1; + switch (node->k) { + case AST_PROC_DEF: + if (scope_add_symbol(scope, node)) + return -1; + break; + + case AST_STRUCT_DEF: + if (scope_add_type(scope, node)) + return -1; + break; + + case AST_STRUCT_CONT: + if (scope_add_chain(scope, node)) + return -1; + break; + + case AST_TRAIT_DEF: + if (scope_add_trait(scope, node)) + return -1; + break; + + default: + abort(); + } } foreach_node(node, root) { @@ -206,6 +206,11 @@ void ast_dump(int depth, struct ast *n) #define DUMP(x) case x: dump(depth, #x); break; switch (n->k) { + DUMP(AST_IMPORT); + DUMP(AST_TRAIT_DEF); + DUMP(AST_TRAIT_APPLY); + DUMP(AST_STRUCT_DEF); + DUMP(AST_STRUCT_CONT); DUMP(AST_CLOSURE); DUMP(AST_IF); DUMP(AST_NIL); @@ -636,6 +641,11 @@ const char *ast_str(enum ast_kind k) { #define CASE(x) case x: return #x; switch (k) { + CASE(AST_IMPORT); + CASE(AST_TRAIT_DEF); + CASE(AST_TRAIT_APPLY); + CASE(AST_STRUCT_DEF); + CASE(AST_STRUCT_CONT); CASE(AST_CLOSURE); CASE(AST_IF); CASE(AST_NIL); diff --git a/src/lexer.l b/src/lexer.l index e11b51e..e6a3f22 100644 --- a/src/lexer.l +++ b/src/lexer.l @@ -131,6 +131,10 @@ STRING \"(\\.|[^"\\])*\" "else" {return ELSE;} "nil" {return NIL;} "own" {return OWN;} +"pub" {return PUB;} + +"import" {return IMPORT;} +"continue" {return CONTINUE;} "error" {return ERROR;} diff --git a/src/parser.y b/src/parser.y index 68725a8..7b2c651 100644 --- a/src/parser.y +++ b/src/parser.y @@ -74,7 +74,10 @@ %token ELSE "else" %token NIL "nil" %token OWN "own" +%token PUB "pub" %token MUT "mut" +%token CONTINUE "continue" +%token IMPORT "import" %token ERROR "error" %token DOT "." %token SCOPE "::" @@ -97,11 +100,16 @@ %left "." "=>" "(" ")" %left "::" -%nterm <node> top unit proc call closure var expr statement body +%nterm <node> top unit proc proc_decl call closure var expr statement body %nterm <node> vars exprs statements closures err trailing_closure %nterm <node> opt_vars opt_exprs opt_statements opt_trailing_closure opt_err opt_error %nterm <node> rev_vars rev_exprs rev_closures rev_statements -%nterm <node> let if nil own unop binop construct +%nterm <node> let if nil own unop binop construct import + +%nterm <node> struct struct_cont trait +%nterm <node> type_param type_params opt_type_params +%nterm <node> behaviour behaviours opt_behaviours +%nterm <node> member members opt_members %nterm <type> type rev_types types opt_types @@ -203,7 +211,9 @@ proc : ID "(" opt_vars ")" body { $$ = gen_proc($[ID], $[opt_vars], NULL, $[body], src_loc(@$)); } - | ID "(" opt_vars ")" ";" { + +proc_decl + : ID "(" opt_vars ")" ";" { $$ = gen_proc($[ID], $[opt_vars], NULL, NULL, src_loc(@$)); } @@ -413,8 +423,80 @@ body $$ = gen_block($2, $3, src_loc(@$)); } +behaviour + : APPLY ";" { $$ = gen_trait_apply($1, src_loc(@$)); } + | proc_decl ";" + | proc + +behaviours + : behaviours behaviour { $$ = $1; $1->n = $2; } + | behaviour + +opt_behaviours + : behaviours + | { $$ = NULL; } + +member + : var ";" + | behaviour + +members + : members member { $$ = $1; $1->n = $2; } + | member + +opt_members + : members + | { $$ = NULL; } + +type_param + : ID ID { + struct type *t = tgen_id($1, src_loc(@1)); + $$ = gen_var($2, t, src_loc(@$)); + } + +type_params + : type_param "," type_params { + $$ = $1; $1->n = $3; + } + | type_param + +opt_type_params + : type_params + | { $$ = NULL; } + +struct + : ID "[" opt_type_params "]" "{" opt_members "}" { + $$ = gen_struct($1, $3, $6, src_loc(@$)); + } + +struct_cont + : "continue" ID "[" opt_type_params "]" "{" opt_behaviours "}" { + $$ = gen_struct_cont($2, $4, $7, src_loc(@$)); + } + +trait + : ID "{" opt_behaviours "}" { + $$ = gen_trait($1, $3, src_loc(@$)); + } + +import + : "import" STRING { + $$ = gen_import($2, src_loc(@$)); + } + top : proc + | proc_decl + | struct + | struct_cont + | import + | trait + | "pub" proc { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } + | "pub" proc_decl { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } + | "pub" struct { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } + | "pub" struct_cont { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } + | "pub" import { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } + | "pub" trait { $$ = $2; ast_set_flags($$, AST_FLAG_PUBLIC); } | error { $$ = gen_empty(src_loc(@$)); parser->failed = true; diff --git a/src/scope.c b/src/scope.c index 367384a..fe35d58 100644 --- a/src/scope.c +++ b/src/scope.c @@ -29,6 +29,8 @@ struct scope *create_scope() scope->number = counter++; scope->symbols = visible_create(16); /* arbitrary number */ + scope->traits = visible_create(1); + scope->types = visible_create(1); return scope; } @@ -43,6 +45,8 @@ void destroy_scope(struct scope *scope) } visible_destroy(&scope->symbols); + visible_destroy(&scope->traits); + visible_destroy(&scope->types); struct scope *prev = scope->children, *cur; if (prev) @@ -54,17 +58,17 @@ void destroy_scope(struct scope *scope) free(scope); } -int scope_add_var(struct scope *scope, struct ast *var) +int scope_add_symbol(struct scope *scope, struct ast *symbol) { - assert(var->k == AST_VAR_DEF); - struct ast **exists = visible_insert(&scope->symbols, var_id(var), var); + assert(symbol->k == AST_VAR_DEF || symbol->k == AST_PROC_DEF); + struct ast **exists = visible_insert(&scope->symbols, symbol->s, symbol); if (!exists) { - internal_error("failed inserting var into scope"); + internal_error("failed inserting symbol into scope"); return -1; } - if (*exists != var) { - semantic_error(scope, var, "var redefined"); + if (*exists != symbol) { + semantic_error(scope, symbol, "symbol redefined"); semantic_info(scope, *exists, "previously here"); return -1; } @@ -72,17 +76,38 @@ int scope_add_var(struct scope *scope, struct ast *var) return 0; } -int scope_add_proc(struct scope *scope, struct ast *proc) +struct ast *scope_find_symbol(struct scope *scope, char *id) +{ + struct ast **v = visible_find(&scope->symbols, id); + if (!v) + return NULL; + + return *v; +} + +struct ast *file_scope_find_symbol(struct scope *scope, char *id) +{ + if (!scope) + return NULL; + + struct ast *found = scope_find_symbol(scope, id); + if (found) + return found; + + return file_scope_find_symbol(scope->parent, id); +} + +int scope_add_type(struct scope *scope, struct ast *type) { - assert(proc->k == AST_PROC_DEF); - struct ast **exists = visible_insert(&scope->symbols, proc_id(proc), proc); + assert(type->k == AST_STRUCT_DEF); + struct ast **exists = visible_insert(&scope->symbols, type->s, type); if (!exists) { - internal_error("failed inserting proc into scope"); + internal_error("failed inserting type into scope"); return -1; } - if (*exists != proc) { - semantic_error(scope, proc, "proc redefined"); + if (*exists != type) { + semantic_error(scope, type, "type redefined"); semantic_info(scope, *exists, "previously here"); return -1; } @@ -90,79 +115,99 @@ int scope_add_proc(struct scope *scope, struct ast *proc) return 0; } -struct ast *scope_find_proc(struct scope *scope, char *id) +int scope_add_chain(struct scope *scope, struct ast *type) { - struct ast **v = visible_find(&scope->symbols, id); - if (!v) - return NULL; + assert(type->k == AST_STRUCT_CONT); + struct ast *exists = file_scope_find_type(scope, type->s); + if (!exists) { + semantic_error(scope, type, "no previous definition"); + return -1; + } - struct ast *n = *v; - assert(n); + assert(exists->k == AST_STRUCT_DEF || exists->k == AST_STRUCT_CONT); + if (ast_flags(exists, AST_FLAG_PUBLIC) || !ast_flags(type, AST_FLAG_PUBLIC)) { + type->chain = exists; + struct ast **v = visible_insert(&scope->types, type->s, type); + if (!v) { + internal_error("failed inserting type into chain"); + return -1; + } - if (n->k != AST_PROC_DEF) - return NULL; + /* overwrite root */ + if (*v != type) + *v = type; - return n; -} - -struct ast *file_scope_find_proc(struct scope *scope, char *id) -{ - struct ast *n = file_scope_find_symbol(scope, id); - if (!n) - return NULL; + return 0; + } - if (n->k != AST_PROC_DEF) - return NULL; + struct ast *next = exists->chain; + while (next->k == AST_STRUCT_CONT && !ast_flags(next, AST_FLAG_PUBLIC)) { + exists = next; + next = next->chain; + } - return n; + exists->chain = type; + type->chain = exists; + return 0; } -struct ast *scope_find_symbol(struct scope *scope, char *id) +struct ast *scope_find_type(struct scope *scope, char *id) { - struct ast **v = visible_find(&scope->symbols, id); + struct ast **v = visible_find(&scope->types, id); if (!v) return NULL; return *v; } -struct ast *file_scope_find_symbol(struct scope *scope, char *id) +struct ast *file_scope_find_type(struct scope *scope, char *id) { if (!scope) return NULL; - struct ast *found = scope_find_symbol(scope, id); + struct ast *found = scope_find_type(scope, id); if (found) return found; - return file_scope_find_symbol(scope->parent, id); + return file_scope_find_type(scope->parent, id); } -struct ast *scope_find_var(struct scope *scope, char *id) +int scope_add_trait(struct scope *scope, struct ast *trait) { - struct ast **v = visible_find(&scope->symbols, id); - if (!v) - return NULL; + assert(trait->k == AST_TRAIT_DEF); + struct ast **exists = visible_insert(&scope->symbols, trait->s, trait); + if (!exists) { + internal_error("failed inserting trait into scope"); + return -1; + } - struct ast *n = *v; - assert(n); + if (*exists != trait) { + semantic_error(scope, trait, "trait redefined"); + semantic_info(scope, *exists, "previously here"); + return -1; + } - if (n->k != AST_VAR_DEF) + return 0; +} +struct ast *scope_find_trait(struct scope *scope, char *id) +{ + struct ast **v = visible_find(&scope->traits, id); + if (!v) return NULL; - return n; + return *v; } -struct ast *file_scope_find_var(struct scope *scope, char *id) +struct ast *file_scope_find_trait(struct scope *scope, char *id) { if (!scope) return NULL; - struct ast *found = scope_find_var(scope, id); + struct ast *found = scope_find_trait(scope, id); if (found) return found; - return file_scope_find_var(scope->parent, id); + return file_scope_find_trait(scope->parent, id); } void scope_add_scope(struct scope *parent, struct scope *child) |