From 7d84ddfa65c5dc5266b21033860613c7c86490c4 Mon Sep 17 00:00:00 2001 From: Felipe Pena Date: Wed, 9 Aug 2023 18:38:12 -0300 Subject: [PATCH 1/5] fix --- vlib/v/ast/types.v | 1 + vlib/v/gen/c/assign.v | 15 ++++++++++++--- vlib/v/gen/c/auto_str_methods.v | 5 +++-- vlib/v/gen/c/cgen.v | 13 +++++++++---- vlib/v/gen/c/dumpexpr.v | 9 ++++++++- vlib/v/gen/c/fn.v | 6 ++++++ vlib/v/gen/c/str.v | 6 +++++- vlib/v/parser/fn.v | 6 ++++++ 8 files changed, 50 insertions(+), 11 deletions(-) diff --git a/vlib/v/ast/types.v b/vlib/v/ast/types.v index 7e00250f8e36d8..f190f09c6f7e08 100644 --- a/vlib/v/ast/types.v +++ b/vlib/v/ast/types.v @@ -113,6 +113,7 @@ pub enum TypeFlag { generic shared_f atomic_f + option_mut_param_t } /* diff --git a/vlib/v/gen/c/assign.v b/vlib/v/gen/c/assign.v index 80b6845a2a336f..836dcaad480e21 100644 --- a/vlib/v/gen/c/assign.v +++ b/vlib/v/gen/c/assign.v @@ -610,10 +610,16 @@ fn (mut g Gen) assign_stmt(node_ ast.AssignStmt) { if op_overloaded { g.op_arg(left, op_expected_left, var_type) } else { - if !is_decl && left.is_auto_deref_var() { + if !is_decl && left.is_auto_deref_var() && !var_type.has_flag(.option) { g.write('*') } - g.expr(left) + if var_type.has_flag(.option_mut_param_t) { + g.write('memcpy(&') + g.expr(left) + g.write('->data, &') + } else { + g.expr(left) + } if !is_decl && var_type.has_flag(.shared_f) { g.write('->val') // don't reset the mutex, just change the value } @@ -630,7 +636,7 @@ fn (mut g Gen) assign_stmt(node_ ast.AssignStmt) { if is_decl { g.writeln(';') } - } else if !g.is_arraymap_set && !str_add && !op_overloaded { + } else if !var_type.has_flag(.option_mut_param_t) && !g.is_arraymap_set && !str_add && !op_overloaded { g.write(' ${op} ') } else if str_add || op_overloaded { g.write(', ') @@ -739,6 +745,9 @@ fn (mut g Gen) assign_stmt(node_ ast.AssignStmt) { if str_add || op_overloaded { g.write(')') } + if var_type.has_flag(.option_mut_param_t) { + g.write('.data, sizeof(${g.base_type(val_type)}))') + } if g.is_arraymap_set { g.write(' })') g.is_arraymap_set = false diff --git a/vlib/v/gen/c/auto_str_methods.v b/vlib/v/gen/c/auto_str_methods.v index 9df74aff80a44d..5d3c47adbe6fb0 100644 --- a/vlib/v/gen/c/auto_str_methods.v +++ b/vlib/v/gen/c/auto_str_methods.v @@ -112,7 +112,8 @@ fn (mut g Gen) final_gen_str(typ StrType) { } g.str_fn_names << str_fn_name if typ.typ.has_flag(.option) { - g.gen_str_for_option(typ.typ, styp, str_fn_name) + opt_typ := if typ.typ.has_flag(.option_mut_param_t) { styp.replace('*', '') } else { styp } + g.gen_str_for_option(typ.typ, opt_typ, str_fn_name) return } if typ.typ.has_flag(.result) { @@ -185,7 +186,7 @@ fn (mut g Gen) gen_str_for_option(typ ast.Type, styp string, str_fn_name string) g.auto_str_funcs.writeln('string indent_${str_fn_name}(${styp} it, int indent_count) {') g.auto_str_funcs.writeln('\tstring res;') g.auto_str_funcs.writeln('\tif (it.state == 0) {') - deref := if typ.is_ptr() { '**(${sym.cname}**)&' } else { '*(${sym.cname}*)' } + deref := if typ.is_ptr() && !typ.has_flag(.option_mut_param_t) { '**(${sym.cname}**)&' } else { '*(${sym.cname}*)' } if sym.kind == .string { if typ.nr_muls() > 1 { g.auto_str_funcs.writeln('\t\tres = ptr_str(*(${sym.cname}**)&it.data);') diff --git a/vlib/v/gen/c/cgen.v b/vlib/v/gen/c/cgen.v index b34886a590dfd8..160f27b786298a 100644 --- a/vlib/v/gen/c/cgen.v +++ b/vlib/v/gen/c/cgen.v @@ -1076,7 +1076,7 @@ fn (mut g Gen) option_type_name(t ast.Type) (string, string) { } else { styp = '${c.option_name}_${base}' } - if t.is_ptr() { + if t.is_ptr() && !t.has_flag(.option_mut_param_t) { styp = styp.replace('*', '_ptr') } return styp, base @@ -1160,7 +1160,7 @@ fn (mut g Gen) write_options() { done = g.done_options.clone() } for base, styp in g.options { - if base in done { + if base in done || styp.ends_with('*') { continue } done << base @@ -1947,7 +1947,12 @@ fn (mut g Gen) expr_with_tmp_var(expr ast.Expr, expr_typ ast.Type, ret_typ ast.T ret_styp := g.typ(g.unwrap_generic(ret_typ)).replace('*', '_ptr') g.writeln('${ret_styp} ${tmp_var};') } else { - g.writeln('${g.typ(ret_typ)} ${tmp_var};') + if ret_typ.has_flag(.option_mut_param_t) { + styp = styp.replace('*', '') + g.writeln('${g.typ(ret_typ).replace('*', '')} ${tmp_var};') + } else { + g.writeln('${g.typ(ret_typ)} ${tmp_var};') + } } if ret_typ.has_flag(.option) { if expr_typ.has_flag(.option) && expr in [ast.StructInit, ast.ArrayInit, ast.MapInit] { @@ -1969,7 +1974,7 @@ fn (mut g Gen) expr_with_tmp_var(expr ast.Expr, expr_typ ast.Type, ret_typ ast.T } else { g.write('_option_ok(&(${styp}[]) { ') } - if !expr_typ.is_ptr() && ret_typ.is_ptr() { + if !expr_typ.is_ptr() && ret_typ.is_ptr() && !ret_typ.has_flag(.option_mut_param_t) { g.write('&/*ref*/') } } diff --git a/vlib/v/gen/c/dumpexpr.v b/vlib/v/gen/c/dumpexpr.v index a49a5a51166838..ec6cecd310ded5 100644 --- a/vlib/v/gen/c/dumpexpr.v +++ b/vlib/v/gen/c/dumpexpr.v @@ -71,6 +71,9 @@ fn (mut g Gen) dump_expr(node ast.DumpExpr) { } else { old_inside_opt_or_res := g.inside_opt_or_res g.inside_opt_or_res = true + if expr_type.has_flag(.option_mut_param_t) { + g.write('*') + } g.expr(node.expr) g.inside_opt_or_res = old_inside_opt_or_res } @@ -106,7 +109,11 @@ fn (mut g Gen) dump_expr_definitions() { } else { if typ.has_flag(.option) { str_dumparg_type += '_option_' - ptr_asterisk = ptr_asterisk.replace('*', '_ptr') + if typ.has_flag(.option_mut_param_t) { + ptr_asterisk = ptr_asterisk.replace('*', '') + } else { + ptr_asterisk = ptr_asterisk.replace('*', '_ptr') + } } str_dumparg_type += g.cc_type(dump_type, true) + ptr_asterisk } diff --git a/vlib/v/gen/c/fn.v b/vlib/v/gen/c/fn.v index 35b07e8149eb13..52064a4e3acb5a 100644 --- a/vlib/v/gen/c/fn.v +++ b/vlib/v/gen/c/fn.v @@ -2210,6 +2210,9 @@ fn (mut g Gen) ref_or_deref_arg(arg ast.CallArg, expected_type ast.Type, lang as && lang != .c { if arg.expr.is_lvalue() { if expected_type.has_flag(.option) { + if expected_type.has_flag(.option_mut_param_t) { + g.write('&/*opt-mut*/') + } g.expr_with_opt(arg.expr, arg_typ, expected_type) return } else { @@ -2250,6 +2253,9 @@ fn (mut g Gen) ref_or_deref_arg(arg ast.CallArg, expected_type ast.Type, lang as g.write('->val') return } else if expected_type.has_flag(.option) { + if expected_type.has_flag(.option_mut_param_t) { + g.write('&/*opt-mut*/') + } g.expr_with_opt(arg.expr, arg_typ, expected_type) return } else if arg.expr is ast.ArrayInit { diff --git a/vlib/v/gen/c/str.v b/vlib/v/gen/c/str.v index e07b732be587c5..fd49e562489821 100644 --- a/vlib/v/gen/c/str.v +++ b/vlib/v/gen/c/str.v @@ -127,7 +127,11 @@ fn (mut g Gen) gen_expr_to_string(expr ast.Expr, etype ast.Type) { if str_method_expects_ptr && !is_ptr { g.write('&') } else if is_ptr && typ.has_flag(.option) { - g.write('*(${g.typ(typ)}*)&') + if typ.has_flag(.option_mut_param_t) { + g.write('*') + } else { + g.write('*(${g.typ(typ)}*)&') + } } else if !str_method_expects_ptr && !is_shared && (is_ptr || is_var_mut) { g.write('*'.repeat(typ.nr_muls())) } diff --git a/vlib/v/parser/fn.v b/vlib/v/parser/fn.v index a16034b4d8a97b..04edd00c2fd5cd 100644 --- a/vlib/v/parser/fn.v +++ b/vlib/v/parser/fn.v @@ -913,6 +913,9 @@ fn (mut p Parser) fn_params() ([]ast.Param, bool, bool) { // p.error('cannot mut') // } // arg_type = arg_type.ref() + if param_type.has_flag(.option) && param_type.nr_muls() == 0 { + param_type = param_type.set_flag(.option_mut_param_t) + } param_type = param_type.set_nr_muls(1) if is_shared { param_type = param_type.set_flag(.shared_f) @@ -1025,6 +1028,9 @@ fn (mut p Parser) fn_params() ([]ast.Param, bool, bool) { pos) return []ast.Param{}, false, false } + if typ.has_flag(.option) && typ.nr_muls() == 0 { + typ = typ.set_flag(.option_mut_param_t) + } typ = typ.set_nr_muls(1) if is_shared { typ = typ.set_flag(.shared_f) From 20ef2354e7ef508960993bf67d19855a507a2a1c Mon Sep 17 00:00:00 2001 From: Felipe Pena Date: Wed, 9 Aug 2023 18:40:57 -0300 Subject: [PATCH 2/5] fmt --- vlib/v/gen/c/assign.v | 5 +++-- vlib/v/gen/c/auto_str_methods.v | 6 +++++- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/vlib/v/gen/c/assign.v b/vlib/v/gen/c/assign.v index 836dcaad480e21..3e99661dac3634 100644 --- a/vlib/v/gen/c/assign.v +++ b/vlib/v/gen/c/assign.v @@ -636,7 +636,8 @@ fn (mut g Gen) assign_stmt(node_ ast.AssignStmt) { if is_decl { g.writeln(';') } - } else if !var_type.has_flag(.option_mut_param_t) && !g.is_arraymap_set && !str_add && !op_overloaded { + } else if !var_type.has_flag(.option_mut_param_t) && !g.is_arraymap_set && !str_add + && !op_overloaded { g.write(' ${op} ') } else if str_add || op_overloaded { g.write(', ') @@ -746,7 +747,7 @@ fn (mut g Gen) assign_stmt(node_ ast.AssignStmt) { g.write(')') } if var_type.has_flag(.option_mut_param_t) { - g.write('.data, sizeof(${g.base_type(val_type)}))') + g.write('.data, sizeof(${g.base_type(val_type)}))') } if g.is_arraymap_set { g.write(' })') diff --git a/vlib/v/gen/c/auto_str_methods.v b/vlib/v/gen/c/auto_str_methods.v index 5d3c47adbe6fb0..b9b0f62b343c0b 100644 --- a/vlib/v/gen/c/auto_str_methods.v +++ b/vlib/v/gen/c/auto_str_methods.v @@ -186,7 +186,11 @@ fn (mut g Gen) gen_str_for_option(typ ast.Type, styp string, str_fn_name string) g.auto_str_funcs.writeln('string indent_${str_fn_name}(${styp} it, int indent_count) {') g.auto_str_funcs.writeln('\tstring res;') g.auto_str_funcs.writeln('\tif (it.state == 0) {') - deref := if typ.is_ptr() && !typ.has_flag(.option_mut_param_t) { '**(${sym.cname}**)&' } else { '*(${sym.cname}*)' } + deref := if typ.is_ptr() && !typ.has_flag(.option_mut_param_t) { + '**(${sym.cname}**)&' + } else { + '*(${sym.cname}*)' + } if sym.kind == .string { if typ.nr_muls() > 1 { g.auto_str_funcs.writeln('\t\tres = ptr_str(*(${sym.cname}**)&it.data);') From c80ac0c49e1c70afbe5faf13fcfff414b90ec202 Mon Sep 17 00:00:00 2001 From: Felipe Pena Date: Sun, 20 Aug 2023 15:36:49 -0300 Subject: [PATCH 3/5] fix --- vlib/v/gen/c/cgen.v | 3 ++- vlib/v/tests/option_mut_param_test.v | 23 +++++++++++++++++++++++ 2 files changed, 25 insertions(+), 1 deletion(-) create mode 100644 vlib/v/tests/option_mut_param_test.v diff --git a/vlib/v/gen/c/cgen.v b/vlib/v/gen/c/cgen.v index 160f27b786298a..eef8094477b62f 100644 --- a/vlib/v/gen/c/cgen.v +++ b/vlib/v/gen/c/cgen.v @@ -1949,7 +1949,8 @@ fn (mut g Gen) expr_with_tmp_var(expr ast.Expr, expr_typ ast.Type, ret_typ ast.T } else { if ret_typ.has_flag(.option_mut_param_t) { styp = styp.replace('*', '') - g.writeln('${g.typ(ret_typ).replace('*', '')} ${tmp_var};') + ret_styp := g.typ(ret_typ).replace('*', '') + g.writeln('${ret_styp} ${tmp_var};') } else { g.writeln('${g.typ(ret_typ)} ${tmp_var};') } diff --git a/vlib/v/tests/option_mut_param_test.v b/vlib/v/tests/option_mut_param_test.v new file mode 100644 index 00000000000000..3c254f375e3585 --- /dev/null +++ b/vlib/v/tests/option_mut_param_test.v @@ -0,0 +1,23 @@ +struct Abc { + a int +} + +fn foo(mut baz ?&Abc) { + baz = Abc{ + a: 3 + } + println(baz) + dump(baz) +} + +fn test_main() { + mut a := ?Abc{ + a: 2 + } + assert a?.a == 2 + dump(a) + foo(mut a) + println('--') + dump(a) + assert a?.a == 3 +} From 675ebc09e5e8bbc9043041012b8af971450b5310 Mon Sep 17 00:00:00 2001 From: Felipe Pena Date: Tue, 22 Aug 2023 09:35:59 -0300 Subject: [PATCH 4/5] fix --- vlib/v/ast/types.v | 2 +- vlib/v/tests/option_mut_param_test.v | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/vlib/v/ast/types.v b/vlib/v/ast/types.v index f190f09c6f7e08..fc85affbbaeb11 100644 --- a/vlib/v/ast/types.v +++ b/vlib/v/ast/types.v @@ -1402,7 +1402,7 @@ pub fn (t &Table) type_to_str_using_aliases(typ Type, import_aliases map[string] nr_muls-- res = 'atomic ' + res } - if nr_muls > 0 && !typ.has_flag(.variadic) { + if nr_muls > 0 && !typ.has_flag(.variadic) && !typ.has_flag(.option_mut_param_t) { res = strings.repeat(`&`, nr_muls) + res } if typ.has_flag(.option) { diff --git a/vlib/v/tests/option_mut_param_test.v b/vlib/v/tests/option_mut_param_test.v index 3c254f375e3585..6fb858b5ad410c 100644 --- a/vlib/v/tests/option_mut_param_test.v +++ b/vlib/v/tests/option_mut_param_test.v @@ -2,7 +2,7 @@ struct Abc { a int } -fn foo(mut baz ?&Abc) { +fn foo(mut baz ?Abc) { baz = Abc{ a: 3 } From ead3ab316c9405ff8cd67c015d8c526747a763e0 Mon Sep 17 00:00:00 2001 From: Felipe Pena Date: Sat, 9 Sep 2023 09:40:05 -0300 Subject: [PATCH 5/5] fix param validation --- vlib/v/checker/check_types.v | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/vlib/v/checker/check_types.v b/vlib/v/checker/check_types.v index d71455129a390e..2dbed30e07a53c 100644 --- a/vlib/v/checker/check_types.v +++ b/vlib/v/checker/check_types.v @@ -276,7 +276,11 @@ fn (mut c Checker) check_expected_call_arg(got ast.Type, expected_ ast.Type, lan if language != .v || expected.is_ptr() == got.is_ptr() || arg.is_mut || arg.expr.is_auto_deref_var() || got.has_flag(.shared_f) || c.table.sym(expected_).kind !in [.array, .map] { - return + if !expected.has_flag(.option) || !arg.is_mut + || (arg.is_mut && expected.has_flag(.option) + && (expected.has_flag(.option_mut_param_t) || (expected.is_ptr() && got.is_ptr()))) { + return + } } } else { got_typ_sym := c.table.sym(c.unwrap_generic(got))