aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/lower.c9
-rw-r--r--src/move.c32
2 files changed, 36 insertions, 5 deletions
diff --git a/src/lower.c b/src/lower.c
index 5578319..b683e0b 100644
--- a/src/lower.c
+++ b/src/lower.c
@@ -309,6 +309,12 @@ static int lower_mark_moved(struct state *state, struct ast *moves)
if (move->k != AST_ID)
continue;
+ if (is_trivially_copyable(move->t))
+ continue;
+
+ if (is_callable(move->t))
+ continue;
+
printf("%s_owned = false;\n", id_str(move));
indent(state);
}
@@ -450,6 +456,9 @@ static int lower_block_vars(struct state *state, struct ast *block)
if (is_trivially_copyable(def->t))
continue;
+ if (is_callable(def->t))
+ continue;
+
if (!populated) {
indent(state);
printf("[[maybe_unused]] bool %s_owned = true",
diff --git a/src/move.c b/src/move.c
index a87d6dd..6ed687b 100644
--- a/src/move.c
+++ b/src/move.c
@@ -234,6 +234,9 @@ static int total_check_single(struct state *state, struct scope *scope)
if (is_trivially_copyable(def->t))
continue;
+ if (is_callable(def->t))
+ continue;
+
struct ast_pair *prev = find_move(state, def);
if (prev)
continue;
@@ -294,10 +297,27 @@ static int mvcheck_call(struct state *state, struct ast *node)
if (!args)
return 0;
+ /* get all 'normal' parameters moved */
+ foreach_node(arg, call_args(node)) {
+ if (arg->k == AST_CLOSURE)
+ continue;
+
+ if (mvcheck(state, arg))
+ return -1;
+ }
+
+ /* check if there are any arguments that might not be moved if function
+ * returns immediately */
+ if (!call_err(node) && total_check_proc(state, node->scope))
+ semantic_info(node->scope, node, "in implicit error branch");
+
+ /* check into closures */
int ret = 0;
struct state group_state = create_state(state);
-
foreach_node(arg, call_args(node)) {
+ if (arg->k != AST_CLOSURE)
+ continue;
+
struct state arg_state = create_state(state);
ret = mvcheck(&arg_state, arg);
@@ -324,10 +344,6 @@ static int mvcheck_call(struct state *state, struct ast *node)
push_up(&err_state);
destroy_state(&err_state);
}
- else if (!ret) {
- if (total_check_proc(state, node->scope))
- semantic_info(node->scope, node, "in implicit error branch");
- }
destroy_state(&group_state);
return ret;
@@ -507,6 +523,12 @@ static int mvcheck_own(struct state *state, struct ast *node)
struct ast *def = file_scope_find_symbol(node->scope, own_id(node));
assert(def);
+ if (is_callable(def->t)) {
+ semantic_error(node->scope, node,
+ "ownership of callable cannot be checked");
+ return -1;
+ }
+
struct rm_move prev = remove_move(state, def);
int ret = mvcheck(state, own_body(node));