checker: check that fns with return values, and matches, do return

pull/6799/head
Delyan Angelov 2020-11-11 17:23:57 +02:00
parent 6271798ce3
commit 8c241cb745
6 changed files with 55 additions and 19 deletions

View File

@ -407,6 +407,8 @@ fn (mut m map) get_and_set(key string, zero voidptr) voidptr {
// Key not found, insert key with zero-value // Key not found, insert key with zero-value
m.set(key, zero) m.set(key, zero)
} }
assert false
return voidptr(0)
} }
// If `key` matches the key of an element in the container, // If `key` matches the key of an element in the container,

View File

@ -325,6 +325,9 @@ fn (mut ch Channel) try_push_priv(src voidptr, no_block bool) ChanState {
} }
} }
} }
// this should not happen
assert false
return .success
} }
[inline] [inline]
@ -494,8 +497,9 @@ fn (mut ch Channel) try_pop_priv(dest voidptr, no_block bool) ChanState {
dest2 = dest dest2 = dest
} }
} }
return .success break
} }
return .success
} }
// Wait `timeout` on any of `channels[i]` until one of them can push (`is_push[i] = true`) or pop (`is_push[i] = false`) // Wait `timeout` on any of `channels[i]` until one of them can push (`is_push[i] = true`) or pop (`is_push[i] = false`)

View File

@ -3357,8 +3357,8 @@ pub fn (mut c Checker) match_expr(mut node ast.MatchExpr) table.Type {
c.match_exprs(mut node, cond_type_sym) c.match_exprs(mut node, cond_type_sym)
c.expected_type = cond_type c.expected_type = cond_type
mut ret_type := table.void_type mut ret_type := table.void_type
mut require_return := false mut nbranches_with_return := 0
mut branch_without_return := false mut nbranches_without_return := 0
for branch in node.branches { for branch in node.branches {
c.stmts(branch.stmts) c.stmts(branch.stmts)
if node.is_expr && branch.stmts.len > 0 { if node.is_expr && branch.stmts.len > 0 {
@ -3392,18 +3392,22 @@ pub fn (mut c Checker) match_expr(mut node ast.MatchExpr) table.Type {
} }
if has_return := c.has_return(branch.stmts) { if has_return := c.has_return(branch.stmts) {
if has_return { if has_return {
require_return = true nbranches_with_return++
} else { } else {
branch_without_return = true nbranches_without_return++
} }
} }
} }
if require_return && branch_without_return { if nbranches_with_return > 0 {
c.returns = false if nbranches_with_return == node.branches.len {
} else { // an exhaustive match, and all branches returned
// if inner if branch has not covered all branches but this one
c.returns = true c.returns = true
} }
if nbranches_without_return > 0 {
// some of the branches did not return
c.returns = false
}
}
// if ret_type != table.void_type { // if ret_type != table.void_type {
// node.is_expr = c.expected_type != table.void_type // node.is_expr = c.expected_type != table.void_type
// node.expected_type = c.expected_type // node.expected_type = c.expected_type
@ -3692,8 +3696,8 @@ pub fn (mut c Checker) if_expr(mut node ast.IfExpr) table.Type {
expr_required := c.expected_type != table.void_type expr_required := c.expected_type != table.void_type
former_expected_type := c.expected_type former_expected_type := c.expected_type
node.typ = table.void_type node.typ = table.void_type
mut require_return := false mut nbranches_with_return := 0
mut branch_without_return := false mut nbranches_without_return := 0
mut should_skip := false // Whether the current branch should be skipped mut should_skip := false // Whether the current branch should be skipped
mut found_branch := false // Whether a matching branch was found- skip the rest mut found_branch := false // Whether a matching branch was found- skip the rest
for i in 0 .. node.branches.len { for i in 0 .. node.branches.len {
@ -3859,18 +3863,26 @@ pub fn (mut c Checker) if_expr(mut node ast.IfExpr) table.Type {
// Also check for returns inside a comp.if's statements, even if its contents aren't parsed // Also check for returns inside a comp.if's statements, even if its contents aren't parsed
if has_return := c.has_return(branch.stmts) { if has_return := c.has_return(branch.stmts) {
if has_return { if has_return {
require_return = true nbranches_with_return++
} else { } else {
branch_without_return = true nbranches_without_return++
} }
} }
} }
if require_return && (!node.has_else || branch_without_return) { if nbranches_with_return > 0 {
c.returns = false if nbranches_with_return == node.branches.len {
} else { // if/else... where all branches returned
// if inner if branch has not covered all branches but this one
c.returns = true c.returns = true
} }
if !node.has_else {
// `if cond { return ... }` means that when cond is false, execution continues
c.returns = false
}
if nbranches_without_return > 0 {
// some of the branches did not return
c.returns = false
}
}
// if only untyped literals were given default to int/f64 // if only untyped literals were given default to int/f64
if node.typ == table.any_int_type { if node.typ == table.any_int_type {
node.typ = table.int_type node.typ = table.int_type

View File

@ -0,0 +1,5 @@
vlib/v/checker/tests/return_missing_match_simple.vv:1:1: error: missing return at end of function `h`
1 | fn h() int {
| ~~~~~~~~~~
2 | a := 1
3 | match a {

View File

@ -0,0 +1,12 @@
fn h() int {
a := 1
match a {
else { println('abc') }
}
println('hello')
}
fn main() {
d := h()
println('h returned: $d')
}

View File

@ -500,6 +500,7 @@ pub fn (mut s Scanner) buffer_scan() token.Token {
} }
return s.all_tokens[cidx] return s.all_tokens[cidx]
} }
return s.new_token(.eof, '', 1)
} }
[inline] [inline]