diff options
author | Kimplul <kimi.h.kuparinen@gmail.com> | 2025-01-06 01:05:21 +0200 |
---|---|---|
committer | Kimplul <kimi.h.kuparinen@gmail.com> | 2025-01-06 01:05:57 +0200 |
commit | 89bac537165bf262594cca343cb45e16a2167145 (patch) | |
tree | f72a80f1624b12fa3b27c6dc6feedc3e06594e20 /src/move.c | |
parent | aec19e55ca32f68536a550f100d3f058b8a93c02 (diff) | |
download | fwd-89bac537165bf262594cca343cb45e16a2167145.tar.gz fwd-89bac537165bf262594cca343cb45e16a2167145.zip |
implement move checking furthermvcheck
+ Enough that examples still compile, but missing references etc.
Diffstat (limited to 'src/move.c')
-rw-r--r-- | src/move.c | 191 |
1 files changed, 170 insertions, 21 deletions
@@ -47,18 +47,23 @@ static void insert_move(struct state *state, struct ast *def, struct ast *use) moved_insert(&state->moved, pair); } +static void merge(struct state *to, struct state *from) +{ + if (moved_len(&from->moved) == 0) + return; + + foreach(moved, n, &from->moved) { + moved_insert(&to->moved, *n); + } +} + static void push_up(struct state *state) { struct state *parent = state->parent; if (!parent) return; - if (moved_len(&state->moved) == 0) - return; - - foreach(moved, n, &state->moved) { - moved_insert(&parent->moved, *n); - } + merge(parent, state); } static int mvcheck(struct state *state, struct ast *node); @@ -82,19 +87,42 @@ static int mvcheck_block(struct state *state, struct ast *node) return mvcheck_list(state, block_body(node)); } -static int mvcheck_list(struct state *state, struct ast *nodes) +static int mvcheck_call(struct state *state, struct ast *node) { - foreach_node(node, nodes) { - if (mvcheck(state, node)) - return -1; - } + if (mvcheck(state, call_expr(node))) + return -1; - return 0; -} + struct ast *args = call_args(node); + if (!args) + return 0; -static int mvcheck_call(struct state *state, struct ast *node) -{ - return mvcheck_list(state, call_args(node)); + int ret = 0; + long prev_group = args->t->group; + struct state group_state = create_state(state); + + foreach_node(arg, call_args(node)) { + struct state arg_state = create_state(state); + ret = mvcheck(&arg_state, arg); + + long group = arg->t->group; + if (prev_group != group) { + /* previous group ended, push states up */ + push_up(&group_state); + + /* something like reset_state would maybe me more clear? */ + destroy_state(&group_state); + group_state = create_state(state); + } + + merge(&group_state, &arg_state); + destroy_state(&arg_state); + prev_group = group; + if (ret) + break; + } + + destroy_state(&group_state); + return ret; } static int mvcheck_closure(struct state *state, struct ast *node) @@ -107,29 +135,148 @@ static int mvcheck_closure(struct state *state, struct ast *node) return ret; } +static void opt_group_left(struct state *state, struct ast *def, + struct ast *node) +{ + if (!def) + return; + + insert_move(state, def, node); + opt_group_left(state, def->ol, node); +} + +static void opt_group_right(struct state *state, struct ast *def, + struct ast *node) +{ + if (!def) + return; + + insert_move(state, def, node); + opt_group_right(state, def->or, node); +} + static int mvcheck_id(struct state *state, struct ast *node) { struct ast *def = file_scope_find_symbol(node->scope, id_str(node)); assert(def); + /* moves only apply to variables, functions are 'eternal' */ + if (def->k != AST_VAR_DEF) + return 0; + struct ast_pair *prev = find_move(state, def); if (prev) { + /* error messages for opt groups could be improved */ move_error(node, prev->use); return -1; } insert_move(state, def, node); + + if (def->or || def->ol) { + /* we're part of an opt group, add all other members as if they + * were moved as well */ + opt_group_left(state, def->ol, node); + opt_group_right(state, def->or, node); + } + + return 0; +} + +static int mvcheck_let(struct state *state, struct ast *node) +{ + return mvcheck(state, let_expr(node)); +} + +static int mvcheck_init(struct state *state, struct ast *node) +{ + return mvcheck_list(state, init_body(node)); +} + +static int mvcheck_if(struct state *state, struct ast *node) +{ + if (mvcheck(state, if_cond(node))) + return -1; + + struct state body_state = create_state(state); + struct state else_state = create_state(state); + + if (mvcheck(&body_state, if_body(node))) { + destroy_state(&body_state); + destroy_state(&else_state); + return -1; + } + + if (mvcheck(&else_state, if_else(node))) { + destroy_state(&body_state); + destroy_state(&else_state); + return -1; + } + + push_up(&body_state); + push_up(&else_state); + + destroy_state(&body_state); + destroy_state(&else_state); + return 0; +} + +static int mvcheck_unop(struct state *state, struct ast *node) +{ + return mvcheck(state, unop_expr(node)); +} + +static int mvcheck_binop(struct state *state, struct ast *node) +{ + if (mvcheck(state, binop_left(node))) + return -1; + + return mvcheck(state, binop_right(node)); +} + +static int mvcheck_comparison(struct state *state, struct ast *node) +{ + if (mvcheck(state, comparison_left(node))) + return -1; + + return mvcheck(state, comparison_right(node)); +} + +static int mvcheck_list(struct state *state, struct ast *nodes) +{ + foreach_node(node, nodes) { + if (mvcheck(state, node)) + return -1; + } + return 0; } static int mvcheck(struct state *state, struct ast *node) { + if (is_unop(node)) + return mvcheck_unop(state, node); + + if (is_binop(node)) + return mvcheck_binop(state, node); + + if (is_comparison(node)) + return mvcheck_comparison(state, node); + switch (node->k) { - case AST_PROC_DEF: return mvcheck_proc (state, node); - case AST_BLOCK: return mvcheck_block (state, node); - case AST_CALL: return mvcheck_call (state, node); - case AST_CLOSURE: return mvcheck_closure (state, node); - case AST_ID: return mvcheck_id (state, node); + case AST_PROC_DEF: return mvcheck_proc (state, node); + case AST_BLOCK: return mvcheck_block (state, node); + case AST_CALL: return mvcheck_call (state, node); + case AST_CLOSURE: return mvcheck_closure (state, node); + case AST_LET: return mvcheck_let (state, node); + case AST_ID: return mvcheck_id (state, node); + case AST_INIT: return mvcheck_init (state, node); + case AST_IF: return mvcheck_if (state, node); + case AST_CONST_INT: + case AST_CONST_STR: + case AST_CONST_FLOAT: + case AST_CONST_CHAR: + return 0; default: break; } @@ -143,6 +290,8 @@ int mvcheck_root(struct ast *root) struct state state = create_state(NULL); if (mvcheck(&state, node)) return -1; + + destroy_state(&state); } return 0; |