From ece73836aac52cdc19666a69ba0979cc060bf925 Mon Sep 17 00:00:00 2001 From: yuyi Date: Sun, 6 Feb 2022 16:52:36 +0800 Subject: [PATCH] cgen: fix error for struct with reference alias field (#13380) --- vlib/v/gen/c/auto_str_methods.v | 3 ++- vlib/v/gen/c/cgen.v | 6 ++++++ vlib/v/gen/c/str.v | 3 +++ vlib/v/tests/struct_with_reference_alias_field.v | 13 +++++++++++++ 4 files changed, 24 insertions(+), 1 deletion(-) create mode 100644 vlib/v/tests/struct_with_reference_alias_field.v diff --git a/vlib/v/gen/c/auto_str_methods.v b/vlib/v/gen/c/auto_str_methods.v index 84fb9db0f5..b13af61b35 100644 --- a/vlib/v/gen/c/auto_str_methods.v +++ b/vlib/v/gen/c/auto_str_methods.v @@ -916,7 +916,8 @@ fn (mut g Gen) gen_str_for_struct(info ast.Struct, styp string, str_fn_name stri funcprefix += 'isnil(it.${c_name(field.name)})' funcprefix += ' ? _SLIT("nil") : ' // struct, floats and ints have a special case through the _str function - if sym.kind != .struct_ && !field.typ.is_int_valptr() && !field.typ.is_float_valptr() { + if sym.kind !in [.struct_, .alias] && !field.typ.is_int_valptr() + && !field.typ.is_float_valptr() { funcprefix += '*' } } diff --git a/vlib/v/gen/c/cgen.v b/vlib/v/gen/c/cgen.v index b974da241f..102903ba12 100644 --- a/vlib/v/gen/c/cgen.v +++ b/vlib/v/gen/c/cgen.v @@ -4530,6 +4530,12 @@ fn (mut g Gen) cast_expr(node ast.CastExpr) { g.gen_optional_error(node.typ, node.expr) } else { g.write('(${cast_label}(') + if sym.kind == .alias && g.table.final_sym(node.typ).kind == .string { + ptr_cnt := node.typ.nr_muls() - node.expr_type.nr_muls() + if ptr_cnt > 0 { + g.write('&'.repeat(ptr_cnt)) + } + } g.expr(node.expr) if node.expr is ast.IntegerLiteral { if node.typ in [ast.u64_type, ast.u32_type, ast.u16_type] { diff --git a/vlib/v/gen/c/str.v b/vlib/v/gen/c/str.v index 604e14d1d4..fc9f1090e5 100644 --- a/vlib/v/gen/c/str.v +++ b/vlib/v/gen/c/str.v @@ -85,6 +85,9 @@ fn (mut g Gen) gen_expr_to_string(expr ast.Expr, etype ast.Type) { g.expr(expr) g.write(')') } else if typ == ast.string_type { + if etype.is_ptr() { + g.write('*') + } g.expr(expr) } else if typ == ast.bool_type { g.expr(expr) diff --git a/vlib/v/tests/struct_with_reference_alias_field.v b/vlib/v/tests/struct_with_reference_alias_field.v new file mode 100644 index 0000000000..564e388750 --- /dev/null +++ b/vlib/v/tests/struct_with_reference_alias_field.v @@ -0,0 +1,13 @@ +type SS = string + +struct ST { + data &SS +} + +fn test_struct_with_reference_alias_fields() { + mut val := ST{ + data: &SS('hi') + } + println(val.data) + assert '$val.data' == 'hi' +}