From f14bd10c00d9bda82ea2a67e29272975e2d3fe0b Mon Sep 17 00:00:00 2001 From: Alexander Medvednikov Date: Fri, 4 Dec 2020 20:44:33 +0100 Subject: [PATCH] autofree: more `return` fixes --- vlib/v/gen/cgen.v | 32 +++++++++++++++++++++----------- vlib/v/parser/assign.v | 2 +- 2 files changed, 22 insertions(+), 12 deletions(-) diff --git a/vlib/v/gen/cgen.v b/vlib/v/gen/cgen.v index 164fa3be6e..1a43329774 100644 --- a/vlib/v/gen/cgen.v +++ b/vlib/v/gen/cgen.v @@ -2089,6 +2089,7 @@ fn (mut g Gen) autofree_scope_vars2(scope &ast.Scope, start_pos int, end_pos int continue } if obj.is_or { + g.writeln('// skipping `or{}` var "$obj.name"') // Skip vars inited with the `or {}`, since they are generated // after the or block in C. continue @@ -4051,12 +4052,16 @@ fn (mut g Gen) return_statement(node ast.Return) { // `return foo(a, b, c)` // `tmp := foo(a, b, c); free(a); free(b); free(c); return tmp;` // Save return value in a temp var so that it all args (a,b,c) can be freed - tmp = g.new_tmp_var() - g.write(g.typ(g.fn_decl.return_type)) - g.write(' ') - g.write(tmp) - g.write(' = ') - // g.write('return $tmp;') + // Don't use a tmp var if a variable is simply returned: `return x` + if node.exprs[0] !is ast.Ident { + tmp = g.new_tmp_var() + g.write(g.typ(g.fn_decl.return_type)) + g.write(' ') + g.write(tmp) + g.write(' = ') + } else { + g.write('return ') + } } else { g.write('return ') } @@ -4069,16 +4074,21 @@ fn (mut g Gen) return_statement(node ast.Return) { g.write(')') } if free { - g.writeln('; // free tmp exprs') - // autofree before `return` - // set free_parent_scopes to true, since all variables defined in parent - // scopes need to be freed before the return expr := node.exprs[0] if expr is ast.Ident { g.returned_var_name = expr.name } + if tmp != '' { + g.writeln('; // free tmp exprs + all vars before return') + } + g.writeln(';') + // autofree before `return` + // set free_parent_scopes to true, since all variables defined in parent + // scopes need to be freed before the return g.autofree_scope_vars(node.pos.pos - 1, node.pos.line_nr, true) - g.write('return $tmp') + if tmp != '' { + g.write('return $tmp') + } } } else { // if node.exprs.len == 0 { println('this should never happen') diff --git a/vlib/v/parser/assign.v b/vlib/v/parser/assign.v index f6980651bc..05e3b7ba4c 100644 --- a/vlib/v/parser/assign.v +++ b/vlib/v/parser/assign.v @@ -148,7 +148,7 @@ fn (mut p Parser) partial_assign_stmt(left []ast.Expr, left_comments []ast.Comme // Set correct variable position (after the or block) // so that autofree doesn't free it in cgen before // it's declared. (`Or` variables are declared after the or block). - if r0.or_block.pos.pos > 0 { + if r0.or_block.pos.pos > 0 && r0.or_block.stmts.len > 0 { v.is_or = true // v.pos = r0.or_block.pos. }