aboutsummaryrefslogtreecommitdiff
path: root/src/analyze.c
diff options
context:
space:
mode:
authorKimplul <kimi.h.kuparinen@gmail.com>2025-01-09 22:26:02 +0200
committerKimplul <kimi.h.kuparinen@gmail.com>2025-01-09 22:26:02 +0200
commit598be4cd1bdd79e4859ae30291f4d65682cc672a (patch)
tree6e7e7ad537214c78049c4b3b2ee694c3b549fa4e /src/analyze.c
parent6f7c2d6daa5c706d441ddc42c5c6409e5266049a (diff)
downloadfwd-598be4cd1bdd79e4859ae30291f4d65682cc672a.tar.gz
fwd-598be4cd1bdd79e4859ae30291f4d65682cc672a.zip
initial reference checking
Diffstat (limited to 'src/analyze.c')
-rw-r--r--src/analyze.c77
1 files changed, 45 insertions, 32 deletions
diff --git a/src/analyze.c b/src/analyze.c
index ab08f01..dd85c08 100644
--- a/src/analyze.c
+++ b/src/analyze.c
@@ -36,8 +36,8 @@ static int analyze_proc(struct state *state, struct scope *scope,
if (analyze_list(&proc_state, proc_scope, proc_params(node)))
return -1;
- struct type *callable = tgen_callable(NULL, node->loc);
- if (!callable) {
+ struct type *proc_type = tgen_func_ptr(NULL, node->loc);
+ if (!proc_type) {
internal_error("failed allocating proc type");
return -1;
}
@@ -53,10 +53,10 @@ static int analyze_proc(struct state *state, struct scope *scope,
/* otherwise same or ending group, don't increment */
param->t->group = group;
- type_append(&callable->t0, clone_type(param->t));
+ type_append(&proc_type->t0, clone_type(param->t));
}
- node->t = callable;
+ node->t = proc_type;
return analyze(&proc_state, proc_scope, proc_body(node));
}
@@ -179,7 +179,6 @@ static int analyze_let(struct state *state, struct scope *scope,
return -1;
}
- /** @todo check move semantics, maybe in another pass? */
node->t = l;
return 0;
}
@@ -249,7 +248,28 @@ static int analyze_call(struct state *state, struct scope *scope,
return -1;
}
- /** @todo check move semantics? */
+ return 0;
+}
+
+static int analyze_ref(struct state *state, struct scope *scope,
+ struct ast *node)
+{
+ struct ast *expr = ref_base(node);
+ if (analyze(state, scope, expr))
+ return -1;
+
+ if (!is_lvalue(expr)) {
+ semantic_error(node->scope, node, "trying to reference rvalue");
+ return -1;
+ }
+
+ struct type *ref = tgen_ref(expr->t, node->loc);
+ if (!ref) {
+ internal_error("failed allocating ref type");
+ return -1;
+ }
+
+ node->t = ref;
return 0;
}
@@ -295,15 +315,17 @@ static int analyze_closure(struct state *state, struct scope *scope,
if (analyze(state, closure_scope, closure_body(node)))
return -1;
- struct type *callable = tgen_callable(NULL, node->loc);
+ struct type *callable = NULL;
+ if (ast_flags(node, AST_FLAG_NOMOVES))
+ callable = tgen_pure_closure(NULL, node->loc);
+ else
+ callable = tgen_closure(NULL, node->loc);
+
if (!callable) {
internal_error("failed allocating closure type");
return -1;
}
- /** @todo use analysis to figure out if this closure can be called
- * multiple times or just once */
-
foreach_node(binding, closure_bindings(node)) {
type_append(&callable->t0, clone_type(binding->t));
}
@@ -417,28 +439,19 @@ static int analyze(struct state *state, struct scope *scope, struct ast *node)
}
switch (node->k) {
- case AST_PROC_DEF: ret = analyze_proc (state, scope, node);
- break;
- case AST_VAR_DEF: ret = analyze_var (state, scope, node);
- break;
- case AST_BLOCK: ret = analyze_block (state, scope, node);
- break;
- case AST_LET: ret = analyze_let (state, scope, node);
- break;
- case AST_INIT: ret = analyze_init (state, scope, node);
- break;
- case AST_CALL: ret = analyze_call (state, scope, node);
- break;
- case AST_ID: ret = analyze_id (state, scope, node);
- break;
- case AST_IF: ret = analyze_if (state, scope, node);
- break;
- case AST_CLOSURE: ret = analyze_closure (state, scope, node);
- break;
- case AST_CONST_INT: ret = analyze_int (state, scope, node);
- break;
- case AST_CONST_STR: ret = analyze_str (state, scope, node);
- break;
+ case AST_CLOSURE: ret = analyze_closure(state, scope, node); break;
+ case AST_PROC_DEF: ret = analyze_proc(state, scope, node); break;
+ case AST_VAR_DEF: ret = analyze_var(state, scope, node); break;
+ case AST_BLOCK: ret = analyze_block(state, scope, node); break;
+ case AST_INIT: ret = analyze_init(state, scope, node); break;
+ case AST_CALL: ret = analyze_call(state, scope, node); break;
+ case AST_REF: ret = analyze_ref(state, scope, node); break;
+ case AST_LET: ret = analyze_let(state, scope, node); break;
+ case AST_ID: ret = analyze_id(state, scope, node); break;
+ case AST_IF: ret = analyze_if(state, scope, node); break;
+
+ case AST_CONST_INT: ret = analyze_int(state, scope, node); break;
+ case AST_CONST_STR: ret = analyze_str(state, scope, node); break;
case AST_EMPTY:
node->t = tgen_void(node->loc);