1
0
mirror of https://github.com/vlang/v.git synced 2023-08-10 21:13:21 +03:00

checker, cgen: fix aliased optional or result fn call (#16908)

This commit is contained in:
yuyi 2023-01-09 02:04:17 +08:00 committed by GitHub
parent ed0500dcfd
commit 241109516f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 83 additions and 19 deletions

View File

@ -923,13 +923,20 @@ fn (mut c Checker) type_implements(typ ast.Type, interface_type ast.Type, pos to
// return the actual type of the expression, once the optional is handled
fn (mut c Checker) check_expr_opt_call(expr ast.Expr, ret_type ast.Type) ast.Type {
if expr is ast.CallExpr {
if expr.return_type.has_flag(.optional) || expr.return_type.has_flag(.result) {
return_modifier_kind := if expr.return_type.has_flag(.optional) {
mut expr_ret_type := expr.return_type
if expr_ret_type != 0 && c.table.sym(expr_ret_type).kind == .alias {
unaliased_ret_type := c.table.unaliased_type(expr_ret_type)
if unaliased_ret_type.has_flag(.optional) || unaliased_ret_type.has_flag(.result) {
expr_ret_type = unaliased_ret_type
}
}
if expr_ret_type.has_flag(.optional) || expr_ret_type.has_flag(.result) {
return_modifier_kind := if expr_ret_type.has_flag(.optional) {
'an option'
} else {
'a result'
}
return_modifier := if expr.return_type.has_flag(.optional) { '?' } else { '!' }
return_modifier := if expr_ret_type.has_flag(.optional) { '?' } else { '!' }
if expr.or_block.kind == .absent {
if c.inside_defer {
c.error('${expr.name}() returns ${return_modifier_kind}, so it should have an `or {}` block at the end',
@ -939,7 +946,7 @@ fn (mut c Checker) check_expr_opt_call(expr ast.Expr, ret_type ast.Type) ast.Typ
expr.pos)
}
} else {
c.check_or_expr(expr.or_block, ret_type, expr.return_type)
c.check_or_expr(expr.or_block, ret_type, expr_ret_type)
}
return ret_type.clear_flag(.optional).clear_flag(.result)
} else if expr.or_block.kind == .block {
@ -2329,6 +2336,12 @@ pub fn (mut c Checker) expr(node_ ast.Expr) ast.Type {
}
ast.CallExpr {
mut ret_type := c.call_expr(mut node)
if ret_type != 0 && c.table.sym(ret_type).kind == .alias {
unaliased_type := c.table.unaliased_type(ret_type)
if unaliased_type.has_flag(.optional) || unaliased_type.has_flag(.result) {
ret_type = unaliased_type
}
}
if !ret_type.has_flag(.optional) && !ret_type.has_flag(.result) {
if node.or_block.kind == .block {
c.error('unexpected `or` block, the function `${node.name}` does neither return an optional nor a result',

View File

@ -12,6 +12,12 @@ fn (mut c Checker) return_stmt(mut node ast.Return) {
}
c.expected_type = c.table.cur_fn.return_type
mut expected_type := c.unwrap_generic(c.expected_type)
if expected_type != 0 && c.table.sym(expected_type).kind == .alias {
unaliased_type := c.table.unaliased_type(expected_type)
if unaliased_type.has_flag(.optional) || unaliased_type.has_flag(.result) {
expected_type = unaliased_type
}
}
expected_type_sym := c.table.sym(expected_type)
if node.exprs.len > 0 && c.table.cur_fn.return_type == ast.void_type {
c.error('unexpected argument, current function does not return anything', node.exprs[0].pos())

View File

@ -0,0 +1,6 @@
vlib/v/checker/tests/aliased_optional_fn_call_err.vv:8:10: error: foo() returns an option, so it should have either an `or {}` block, or `?` at the end
6 |
7 | fn main(){
8 | println(foo())
| ~~~~~
9 | }

View File

@ -0,0 +1,9 @@
type OptStr = ?string
fn foo() OptStr{
return 'abc'
}
fn main(){
println(foo())
}

View File

@ -4485,14 +4485,21 @@ fn (mut g Gen) return_stmt(node ast.Return) {
// got to do a correct check for multireturn
sym := g.table.sym(g.fn_decl.return_type)
mut fn_ret_type := g.fn_decl.return_type
if sym.kind == .alias {
unaliased_type := g.table.unaliased_type(fn_ret_type)
if unaliased_type.has_flag(.optional) || unaliased_type.has_flag(.result) {
fn_ret_type = unaliased_type
}
}
fn_return_is_multi := sym.kind == .multi_return
fn_return_is_optional := g.fn_decl.return_type.has_flag(.optional)
fn_return_is_result := g.fn_decl.return_type.has_flag(.result)
fn_return_is_optional := fn_ret_type.has_flag(.optional)
fn_return_is_result := fn_ret_type.has_flag(.result)
mut has_semicolon := false
if node.exprs.len == 0 {
g.write_defer_stmts_when_needed()
if fn_return_is_optional || fn_return_is_result {
styp := g.typ(g.fn_decl.return_type)
styp := g.typ(fn_ret_type)
g.writeln('return (${styp}){0};')
} else {
if g.is_autofree {
@ -4504,7 +4511,7 @@ fn (mut g Gen) return_stmt(node ast.Return) {
return
}
tmpvar := g.new_tmp_var()
ret_typ := g.typ(g.unwrap_generic(g.fn_decl.return_type))
ret_typ := g.typ(g.unwrap_generic(fn_ret_type))
mut use_tmp_var := g.defer_stmts.len > 0 || g.defer_profile_code.len > 0
|| g.cur_lock.lockeds.len > 0
// handle promoting none/error/function returning _option'
@ -4516,7 +4523,7 @@ fn (mut g Gen) return_stmt(node ast.Return) {
if g.fn_decl != unsafe { nil } && g.fn_decl.is_test {
test_error_var := g.new_tmp_var()
g.write('${ret_typ} ${test_error_var} = ')
g.gen_optional_error(g.fn_decl.return_type, node.exprs[0])
g.gen_optional_error(fn_ret_type, node.exprs[0])
g.writeln(';')
g.write_defer_stmts_when_needed()
g.gen_failing_return_error_for_test_fn(node, test_error_var)
@ -4527,7 +4534,7 @@ fn (mut g Gen) return_stmt(node ast.Return) {
} else {
g.write('return ')
}
g.gen_optional_error(g.fn_decl.return_type, node.exprs[0])
g.gen_optional_error(fn_ret_type, node.exprs[0])
g.writeln(';')
if use_tmp_var {
g.write_defer_stmts_when_needed()
@ -4544,7 +4551,7 @@ fn (mut g Gen) return_stmt(node ast.Return) {
if g.fn_decl != unsafe { nil } && g.fn_decl.is_test {
test_error_var := g.new_tmp_var()
g.write('${ret_typ} ${test_error_var} = ')
g.gen_result_error(g.fn_decl.return_type, node.exprs[0])
g.gen_result_error(fn_ret_type, node.exprs[0])
g.writeln(';')
g.write_defer_stmts_when_needed()
g.gen_failing_return_error_for_test_fn(node, test_error_var)
@ -4555,7 +4562,7 @@ fn (mut g Gen) return_stmt(node ast.Return) {
} else {
g.write('return ')
}
g.gen_result_error(g.fn_decl.return_type, node.exprs[0])
g.gen_result_error(fn_ret_type, node.exprs[0])
g.writeln(';')
if use_tmp_var {
g.write_defer_stmts_when_needed()
@ -4686,16 +4693,16 @@ fn (mut g Gen) return_stmt(node ast.Return) {
}
}
if fn_return_is_optional && !expr_type_is_opt && return_sym.name != c.option_name {
styp := g.base_type(g.fn_decl.return_type)
styp := g.base_type(fn_ret_type)
g.writeln('${ret_typ} ${tmpvar};')
g.write('_option_ok(&(${styp}[]) { ')
if !g.fn_decl.return_type.is_ptr() && node.types[0].is_ptr() {
if !fn_ret_type.is_ptr() && node.types[0].is_ptr() {
if !(node.exprs[0] is ast.Ident && !g.is_amp) {
g.write('*')
}
}
for i, expr in node.exprs {
g.expr_with_cast(expr, node.types[i], g.fn_decl.return_type.clear_flag(.optional).clear_flag(.result))
g.expr_with_cast(expr, node.types[i], fn_ret_type.clear_flag(.optional).clear_flag(.result))
if i < node.exprs.len - 1 {
g.write(', ')
}
@ -4715,16 +4722,16 @@ fn (mut g Gen) return_stmt(node ast.Return) {
}
}
if fn_return_is_result && !expr_type_is_result && return_sym.name != c.result_name {
styp := g.base_type(g.fn_decl.return_type)
styp := g.base_type(fn_ret_type)
g.writeln('${ret_typ} ${tmpvar};')
g.write('_result_ok(&(${styp}[]) { ')
if !g.fn_decl.return_type.is_ptr() && node.types[0].is_ptr() {
if !fn_ret_type.is_ptr() && node.types[0].is_ptr() {
if !(node.exprs[0] is ast.Ident && !g.is_amp) {
g.write('*')
}
}
for i, expr in node.exprs {
g.expr_with_cast(expr, node.types[i], g.fn_decl.return_type.clear_flag(.result))
g.expr_with_cast(expr, node.types[i], fn_ret_type.clear_flag(.result))
if i < node.exprs.len - 1 {
g.write(', ')
}

View File

@ -684,6 +684,12 @@ fn (mut g Gen) call_expr(node ast.CallExpr) {
tmp_opt := if gen_or || gen_keep_alive { g.new_tmp_var() } else { '' }
if gen_or || gen_keep_alive {
mut ret_typ := node.return_type
if g.table.sym(ret_typ).kind == .alias {
unaliased_type := g.table.unaliased_type(ret_typ)
if unaliased_type.has_flag(.optional) || unaliased_type.has_flag(.result) {
ret_typ = unaliased_type
}
}
styp := g.typ(ret_typ)
if gen_or && !is_gen_or_and_assign_rhs {
cur_line = g.go_before_stmt(0)
@ -713,7 +719,13 @@ fn (mut g Gen) call_expr(node ast.CallExpr) {
}
if gen_or {
g.or_block(tmp_opt, node.or_block, node.return_type)
unwrapped_typ := node.return_type.clear_flag(.optional).clear_flag(.result)
mut unwrapped_typ := node.return_type.clear_flag(.optional).clear_flag(.result)
if g.table.sym(unwrapped_typ).kind == .alias {
unaliased_type := g.table.unaliased_type(unwrapped_typ)
if unaliased_type.has_flag(.optional) || unaliased_type.has_flag(.result) {
unwrapped_typ = unaliased_type.clear_flag(.optional).clear_flag(.result)
}
}
unwrapped_styp := g.typ(unwrapped_typ)
if g.infix_left_var_name.len > 0 {
g.indent--

View File

@ -0,0 +1,11 @@
type OptStr = ?string
fn foo() OptStr {
return 'abc'
}
fn test_aliased_optional_fn_call() {
ret := foo()?
println(ret)
assert ret == 'abc'
}