compiler: propagate optional

pull/4997/head
Enzo Baldisserri 2020-05-23 08:51:15 +02:00 committed by GitHub
parent 5037d9de37
commit 801bca1ef2
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 184 additions and 80 deletions

View File

@ -720,11 +720,17 @@ pub mut:
expr_type table.Type expr_type table.Type
} }
pub enum OrKind {
absent
block
propagate
}
// `or { ... }` // `or { ... }`
pub struct OrExpr { pub struct OrExpr {
pub: pub:
stmts []Stmt stmts []Stmt
is_used bool // if the or{} block is written down or left out kind OrKind
} }
pub struct Assoc { pub struct Assoc {

View File

@ -24,7 +24,7 @@ pub mut:
warnings []errors.Warning warnings []errors.Warning
error_lines []int // to avoid printing multiple errors for the same line error_lines []int // to avoid printing multiple errors for the same line
expected_type table.Type expected_type table.Type
fn_return_type table.Type // current function's return type cur_fn &ast.FnDecl // current function
const_decl string const_decl string
const_deps []string const_deps []string
const_names []string const_names []string
@ -44,6 +44,7 @@ pub fn new_checker(table &table.Table, pref &pref.Preferences) Checker {
return Checker{ return Checker{
table: table table: table
pref: pref pref: pref
cur_fn: 0
} }
} }
@ -1014,24 +1015,34 @@ fn (mut c Checker) type_implements(typ, inter_typ table.Type, pos token.Position
} }
} }
pub fn (mut c Checker) check_expr_opt_call(expr ast.Expr, xtype table.Type, is_return_used bool) { pub fn (mut c Checker) check_expr_opt_call(expr ast.Expr, ret_type table.Type, is_return_used bool) {
if expr is ast.CallExpr { if expr is ast.CallExpr {
call_expr := expr as ast.CallExpr call_expr := expr as ast.CallExpr
if call_expr.return_type.flag_is(.optional) { if call_expr.return_type.flag_is(.optional) {
c.check_or_block(call_expr, xtype, is_return_used) c.check_or_block(call_expr, ret_type, is_return_used)
} else if call_expr.or_block.is_used { } else if call_expr.or_block.kind == .block {
c.error('unexpected `or` block, the function `$call_expr.name` does not return an optional', c.error('unexpected `or` block, the function `$call_expr.name` does not return an optional',
call_expr.pos) call_expr.pos)
} else if call_expr.or_block.kind == .propagate {
c.error('unexpected `?`, the function `$call_expr.name`, does not return an optional',
call_expr.pos)
} }
} }
} }
pub fn (mut c Checker) check_or_block(mut call_expr ast.CallExpr, ret_type table.Type, is_ret_used bool) { pub fn (mut c Checker) check_or_block(mut call_expr ast.CallExpr, ret_type table.Type, is_ret_used bool) {
if !call_expr.or_block.is_used { if call_expr.or_block.kind == .absent {
c.error('${call_expr.name}() returns an option, but you missed to add an `or {}` block to it', c.error('${call_expr.name}() returns an option, but you missed to add an `or {}` block to it',
call_expr.pos) call_expr.pos)
return return
} }
if call_expr.or_block.kind == .propagate {
if !c.cur_fn.return_type.flag_is(.optional) && c.cur_fn.name != 'main' {
c.error('to propagate the optional call, `${c.cur_fn.name}` must itself return an optional',
call_expr.pos)
}
return
}
stmts_len := call_expr.or_block.stmts.len stmts_len := call_expr.or_block.stmts.len
if stmts_len == 0 { if stmts_len == 0 {
if is_ret_used { if is_ret_used {
@ -1044,7 +1055,7 @@ pub fn (mut c Checker) check_or_block(mut call_expr ast.CallExpr, ret_type table
} }
last_stmt := call_expr.or_block.stmts[stmts_len - 1] last_stmt := call_expr.or_block.stmts[stmts_len - 1]
if is_ret_used { if is_ret_used {
if !c.is_last_or_block_stmt_valid(last_stmt) { if !(last_stmt is ast.Return || last_stmt is ast.BranchStmt || last_stmt is ast.ExprStmt) {
expected_type_name := c.table.get_type_symbol(ret_type).name expected_type_name := c.table.get_type_symbol(ret_type).name
c.error('last statement in the `or {}` block should return `$expected_type_name`', c.error('last statement in the `or {}` block should return `$expected_type_name`',
call_expr.pos) call_expr.pos)
@ -1084,16 +1095,6 @@ fn is_expr_panic_or_exit(expr ast.Expr) bool {
} }
} }
// TODO: merge to check_or_block when v can handle it
pub fn (mut c Checker) is_last_or_block_stmt_valid(stmt ast.Stmt) bool {
return match stmt {
ast.Return { true }
ast.BranchStmt { true }
ast.ExprStmt { true }
else { false }
}
}
pub fn (mut c Checker) selector_expr(mut selector_expr ast.SelectorExpr) table.Type { pub fn (mut c Checker) selector_expr(mut selector_expr ast.SelectorExpr) table.Type {
typ := c.expr(selector_expr.expr) typ := c.expr(selector_expr.expr)
if typ == table.void_type_idx { if typ == table.void_type_idx {
@ -1126,19 +1127,19 @@ pub fn (mut c Checker) selector_expr(mut selector_expr ast.SelectorExpr) table.T
// TODO: non deferred // TODO: non deferred
pub fn (mut c Checker) return_stmt(mut return_stmt ast.Return) { pub fn (mut c Checker) return_stmt(mut return_stmt ast.Return) {
c.expected_type = c.fn_return_type c.expected_type = c.cur_fn.return_type
if return_stmt.exprs.len > 0 && c.fn_return_type == table.void_type { if return_stmt.exprs.len > 0 && c.expected_type == table.void_type {
c.error('too many arguments to return, current function does not return anything', c.error('too many arguments to return, current function does not return anything',
return_stmt.pos) return_stmt.pos)
return return
} else if return_stmt.exprs.len == 0 && c.fn_return_type != table.void_type { } else if return_stmt.exprs.len == 0 && c.expected_type != table.void_type {
c.error('too few arguments to return', return_stmt.pos) c.error('too few arguments to return', return_stmt.pos)
return return
} }
if return_stmt.exprs.len == 0 { if return_stmt.exprs.len == 0 {
return return
} }
expected_type := c.fn_return_type expected_type := c.expected_type
expected_type_sym := c.table.get_type_symbol(expected_type) expected_type_sym := c.table.get_type_symbol(expected_type)
exp_is_optional := expected_type.flag_is(.optional) exp_is_optional := expected_type.flag_is(.optional)
mut expected_types := [expected_type] mut expected_types := [expected_type]
@ -1646,10 +1647,10 @@ fn (mut c Checker) stmts(stmts []ast.Stmt) {
pub fn (mut c Checker) expr(node ast.Expr) table.Type { pub fn (mut c Checker) expr(node ast.Expr) table.Type {
match mut node { match mut node {
ast.AnonFn { ast.AnonFn {
keep_ret_type := c.fn_return_type keep_fn := c.cur_fn
c.fn_return_type = it.decl.return_type c.cur_fn = &it.decl
c.stmts(it.decl.stmts) c.stmts(it.decl.stmts)
c.fn_return_type = keep_ret_type c.cur_fn = keep_fn
return it.typ return it.typ
} }
ast.ArrayInit { ast.ArrayInit {
@ -2361,7 +2362,7 @@ fn (mut c Checker) fn_decl(it ast.FnDecl) {
} }
} }
c.expected_type = table.void_type c.expected_type = table.void_type
c.fn_return_type = it.return_type c.cur_fn = &it
c.stmts(it.stmts) c.stmts(it.stmts)
if it.language == .v && !it.no_body && it.return_type != table.void_type && !c.returns && if it.language == .v && !it.no_body && it.return_type != table.void_type && !c.returns &&
it.name !in ['panic', 'exit'] { it.name !in ['panic', 'exit'] {

View File

@ -0,0 +1,6 @@
vlib/v/checker/tests/unexpected_or.v:6:7: error: unexpected `or` block, the function `ret_zero` does not return an optional
4 |
5 | fn main() {
6 | _ := ret_zero() or { 1 }
| ~~~~~~~~~~
7 | }

View File

@ -0,0 +1,7 @@
fn ret_zero() int {
return 0
}
fn main() {
_ := ret_zero() or { 1 }
}

View File

@ -0,0 +1,7 @@
vlib/v/checker/tests/unexpected_or_propagate.v:6:7: error: unexpected `?`, the function `ret_zero`, does not return an optional
4 |
5 | fn opt_fn() ?int {
6 | a := ret_zero()?
| ~~~~~~~~~~
7 | return a
8 | }

View File

@ -0,0 +1,12 @@
fn ret_zero() int {
return 0
}
fn opt_fn() ?int {
a := ret_zero()?
return a
}
fn main() {
opt_fn() or {}
}

View File

@ -0,0 +1,7 @@
vlib/v/checker/tests/wrong_propagate_ret_type.v:6:7: error: to propagate the optional call, `opt_call` must itself return an optional
4 |
5 | fn opt_call() int {
6 | a := ret_none()?
| ~~~~~~~~~~
7 | return a
8 | }

View File

@ -0,0 +1,8 @@
fn ret_none() ?int {
return none
}
fn opt_call() int {
a := ret_none()?
return a
}

View File

@ -725,10 +725,16 @@ pub fn (mut f Fmt) call_args(args []ast.CallArg) {
} }
pub fn (mut f Fmt) or_expr(or_block ast.OrExpr) { pub fn (mut f Fmt) or_expr(or_block ast.OrExpr) {
if or_block.is_used { match or_block.kind {
f.writeln(' or {') .absent {}
f.stmts(or_block.stmts) .block {
f.write('}') f.writeln(' or {')
f.stmts(or_block.stmts)
f.write('}')
}
.propagate {
f.write('?')
}
} }
} }

View File

@ -0,0 +1,3 @@
fn opt_propagate() ?int {
eventual_wrong_int()?
}

View File

@ -95,6 +95,7 @@ mut:
is_builtin_mod bool is_builtin_mod bool
hotcode_fn_names []string hotcode_fn_names []string
fn_main &ast.FnDecl // the FnDecl of the main function. Needed in order to generate the main function code *last* fn_main &ast.FnDecl // the FnDecl of the main function. Needed in order to generate the main function code *last*
cur_fn &ast.FnDecl
cur_generic_type table.Type // `int`, `string`, etc in `foo<T>()` cur_generic_type table.Type // `int`, `string`, etc in `foo<T>()`
} }
@ -124,6 +125,7 @@ pub fn cgen(files []ast.File, table &table.Table, pref &pref.Preferences) string
pref: pref pref: pref
fn_decl: 0 fn_decl: 0
fn_main: 0 fn_main: 0
cur_fn: 0
autofree: true autofree: true
indent: -1 indent: -1
module_built: pref.path.after('vlib/') module_built: pref.path.after('vlib/')
@ -895,7 +897,6 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
1 { 1 {
// multi return // multi return
// TODO Handle in if_expr // TODO Handle in if_expr
mut or_stmts := []ast.Stmt{}
is_optional := return_type.flag_is(.optional) is_optional := return_type.flag_is(.optional)
mr_var_name := 'mr_$assign_stmt.pos.pos' mr_var_name := 'mr_$assign_stmt.pos.pos'
mr_styp := g.typ(return_type) mr_styp := g.typ(return_type)
@ -905,9 +906,8 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
g.is_assign_rhs = false g.is_assign_rhs = false
if is_optional && assign_stmt.right[0] is ast.CallExpr { if is_optional && assign_stmt.right[0] is ast.CallExpr {
val := assign_stmt.right[0] as ast.CallExpr val := assign_stmt.right[0] as ast.CallExpr
or_stmts = val.or_block.stmts
return_type = val.return_type return_type = val.return_type
g.or_block(mr_var_name, or_stmts, return_type) g.or_block(mr_var_name, val.or_block, return_type)
} }
g.writeln(';') g.writeln(';')
for i, ident in assign_stmt.left { for i, ident in assign_stmt.left {
@ -936,12 +936,10 @@ fn (mut g Gen) gen_assign_stmt(assign_stmt ast.AssignStmt) {
ident_var_info := ident.var_info() ident_var_info := ident.var_info()
styp := g.typ(ident_var_info.typ) styp := g.typ(ident_var_info.typ)
mut is_call := false mut is_call := false
mut or_stmts := []ast.Stmt{}
blank_assign := ident.kind == .blank_ident blank_assign := ident.kind == .blank_ident
match val { match val {
ast.CallExpr { ast.CallExpr {
is_call = true is_call = true
or_stmts = it.or_block.stmts
return_type = it.return_type return_type = it.return_type
} }
// TODO: no buffer fiddling // TODO: no buffer fiddling
@ -1454,12 +1452,12 @@ fn (mut g Gen) enum_expr(node ast.Expr) {
fn (mut g Gen) assign_expr(node ast.AssignExpr) { fn (mut g Gen) assign_expr(node ast.AssignExpr) {
// g.write('/*assign_expr*/') // g.write('/*assign_expr*/')
mut is_call := false mut is_call := false
mut or_stmts := []ast.Stmt{} mut or_block := ast.OrExpr{}
mut return_type := table.void_type mut return_type := table.void_type
match node.val { match node.val {
ast.CallExpr { ast.CallExpr {
is_call = true is_call = true
or_stmts = it.or_block.stmts or_block = it.or_block
return_type = it.return_type return_type = it.return_type
} }
else {} else {}
@ -1538,7 +1536,7 @@ fn (mut g Gen) assign_expr(node ast.AssignExpr) {
} }
if gen_or { if gen_or {
// g.write('/*777 $tmp_opt*/') // g.write('/*777 $tmp_opt*/')
g.or_block(tmp_opt, or_stmts, return_type) g.or_block(tmp_opt, or_block, return_type)
unwrapped_type_str := g.typ(return_type.set_flag(.unset)) unwrapped_type_str := g.typ(return_type.set_flag(.unset))
ident := node.left as ast.Ident ident := node.left as ast.Ident
if ident.kind != .blank_ident && ident.info is ast.IdentVar { if ident.kind != .blank_ident && ident.info is ast.IdentVar {
@ -2936,40 +2934,49 @@ fn (mut g Gen) insert_before_stmt(s string) {
// to access its fields (`.ok`, `.error` etc) // to access its fields (`.ok`, `.error` etc)
// `os.cp(...)` => `Option bool tmp = os__cp(...); if (!tmp.ok) { ... }` // `os.cp(...)` => `Option bool tmp = os__cp(...); if (!tmp.ok) { ... }`
// Returns the type of the last stmt // Returns the type of the last stmt
fn (mut g Gen) or_block(var_name string, stmts []ast.Stmt, return_type table.Type) { fn (mut g Gen) or_block(var_name string, or_block ast.OrExpr, return_type table.Type) {
cvar_name := c_name(var_name) cvar_name := c_name(var_name)
mr_styp := g.base_type(return_type) mr_styp := g.base_type(return_type)
g.writeln(';') // or') g.writeln(';') // or')
g.writeln('if (!${cvar_name}.ok) {') g.writeln('if (!${cvar_name}.ok) {')
g.writeln('\tstring err = ${cvar_name}.v_error;') if or_block.kind == .block {
g.writeln('\tint errcode = ${cvar_name}.ecode;') g.writeln('\tstring err = ${cvar_name}.v_error;')
if stmts.len > 0 && stmts[stmts.len - 1] is ast.ExprStmt && (stmts[stmts.len - 1] as ast.ExprStmt).typ != g.writeln('\tint errcode = ${cvar_name}.ecode;')
table.void_type { stmts := or_block.stmts
g.indent++ if stmts.len > 0 && stmts[or_block.stmts.len - 1] is ast.ExprStmt && (stmts[stmts.len -
for i, stmt in stmts { 1] as ast.ExprStmt).typ != table.void_type {
if i == stmts.len - 1 { g.indent++
expr_stmt := stmt as ast.ExprStmt for i, stmt in stmts {
g.stmt_path_pos << g.out.len if i == stmts.len - 1 {
g.write('*(${mr_styp}*) ${cvar_name}.data = ') expr_stmt := stmt as ast.ExprStmt
is_opt_call := expr_stmt.expr is ast.CallExpr && expr_stmt.typ.flag_is(.optional) g.stmt_path_pos << g.out.len
if is_opt_call { g.write('*(${mr_styp}*) ${cvar_name}.data = ')
g.write('*(${mr_styp}*) ') is_opt_call := expr_stmt.expr is ast.CallExpr && expr_stmt.typ.flag_is(.optional)
if is_opt_call {
g.write('*(${mr_styp}*) ')
}
g.expr(expr_stmt.expr)
if is_opt_call {
g.write('.data')
}
if g.inside_ternary == 0 && !(expr_stmt.expr is ast.IfExpr) {
g.writeln(';')
}
g.stmt_path_pos.delete(g.stmt_path_pos.len - 1)
} else {
g.stmt(stmt)
} }
g.expr(expr_stmt.expr)
if is_opt_call {
g.write('.data')
}
if g.inside_ternary == 0 && !(expr_stmt.expr is ast.IfExpr) {
g.writeln(';')
}
g.stmt_path_pos.delete(g.stmt_path_pos.len - 1)
} else {
g.stmt(stmt)
} }
g.indent--
} else {
g.stmts(stmts)
}
} else if or_block.kind == .propagate {
if g.file.mod.name == 'main' && g.cur_fn.name == 'main' {
g.writeln('\tv_panic(${cvar_name}.v_error);')
} else {
g.writeln('\treturn $cvar_name;')
} }
g.indent--
} else {
g.stmts(stmts)
} }
g.write('}') g.write('}')
} }

View File

@ -12,6 +12,11 @@ fn (mut g Gen) gen_fn_decl(it ast.FnDecl) {
// || it.no_body { // || it.no_body {
return return
} }
former_cur_fn := g.cur_fn
g.cur_fn = &it
defer {
g.cur_fn = former_cur_fn
}
is_main := it.name == 'main' is_main := it.name == 'main'
if it.is_generic && g.cur_generic_type == 0 { // need the cur_generic_type check to avoid inf. recursion if it.is_generic && g.cur_generic_type == 0 { // need the cur_generic_type check to avoid inf. recursion
// loop thru each generic type and generate a function // loop thru each generic type and generate a function
@ -284,7 +289,7 @@ fn (mut g Gen) call_expr(node ast.CallExpr) {
if node.should_be_skipped { if node.should_be_skipped {
return return
} }
gen_or := node.or_block.stmts.len > 0 gen_or := node.or_block.kind != .absent
cur_line := if gen_or && g.is_assign_rhs { cur_line := if gen_or && g.is_assign_rhs {
line := g.go_before_stmt(0) line := g.go_before_stmt(0)
g.out.write(tabs[g.indent]) g.out.write(tabs[g.indent])
@ -303,7 +308,7 @@ fn (mut g Gen) call_expr(node ast.CallExpr) {
g.fn_call(node) g.fn_call(node)
} }
if gen_or { if gen_or {
g.or_block(tmp_opt, node.or_block.stmts, node.return_type) g.or_block(tmp_opt, node.or_block, node.return_type)
g.write('\n${cur_line}${tmp_opt}') g.write('\n${cur_line}${tmp_opt}')
} }
} }

View File

@ -19,11 +19,11 @@ pub fn (mut p Parser) call_expr(language table.Language, mod string) ast.CallExp
} else { } else {
p.check_name() p.check_name()
} }
mut is_or_block_used := false mut or_kind := ast.OrKind.absent
if fn_name == 'json.decode' { if fn_name == 'json.decode' {
p.expecting_type = true // Makes name_expr() parse the type `User` in `json.decode(User, txt)` p.expecting_type = true // Makes name_expr() parse the type `User` in `json.decode(User, txt)`
p.expr_mod = '' p.expr_mod = ''
is_or_block_used = true or_kind = .block
} }
mut generic_type := table.void_type mut generic_type := table.void_type
if p.tok.kind == .lt { if p.tok.kind == .lt {
@ -42,9 +42,9 @@ pub fn (mut p Parser) call_expr(language table.Language, mod string) ast.CallExp
pos: first_pos.pos pos: first_pos.pos
len: last_pos.pos - first_pos.pos + last_pos.len len: last_pos.pos - first_pos.pos + last_pos.len
} }
// `foo() or {}``
mut or_stmts := []ast.Stmt{} mut or_stmts := []ast.Stmt{}
if p.tok.kind == .key_orelse { if p.tok.kind == .key_orelse {
// `foo() or {}``
was_inside_or_expr := p.inside_or_expr was_inside_or_expr := p.inside_or_expr
p.inside_or_expr = true p.inside_or_expr = true
p.next() p.next()
@ -61,7 +61,7 @@ pub fn (mut p Parser) call_expr(language table.Language, mod string) ast.CallExp
pos: p.tok.position() pos: p.tok.position()
is_used: true is_used: true
}) })
is_or_block_used = true or_kind = .block
or_stmts = p.parse_block_no_scope() or_stmts = p.parse_block_no_scope()
p.close_scope() p.close_scope()
p.inside_or_expr = was_inside_or_expr p.inside_or_expr = was_inside_or_expr
@ -69,10 +69,7 @@ pub fn (mut p Parser) call_expr(language table.Language, mod string) ast.CallExp
if p.tok.kind == .question { if p.tok.kind == .question {
// `foo()?` // `foo()?`
p.next() p.next()
is_or_block_used = true or_kind = .propagate
// mut s := ast.Stmt{}
// s = ast.ReturnStmt{}
or_stmts << ast.Return{}
} }
node := ast.CallExpr{ node := ast.CallExpr{
name: fn_name name: fn_name
@ -82,7 +79,7 @@ pub fn (mut p Parser) call_expr(language table.Language, mod string) ast.CallExp
language: language language: language
or_block: ast.OrExpr{ or_block: ast.OrExpr{
stmts: or_stmts stmts: or_stmts
is_used: is_or_block_used kind: or_kind
} }
generic_type: generic_type generic_type: generic_type
} }

View File

@ -986,7 +986,7 @@ fn (mut p Parser) dot_expr(left ast.Expr) ast.Expr {
} }
p.check(.rpar) p.check(.rpar)
mut or_stmts := []ast.Stmt{} mut or_stmts := []ast.Stmt{}
mut is_or_block_used := false mut or_kind := ast.OrKind.absent
if p.tok.kind == .key_orelse { if p.tok.kind == .key_orelse {
p.next() p.next()
p.open_scope() p.open_scope()
@ -1002,10 +1002,15 @@ fn (mut p Parser) dot_expr(left ast.Expr) ast.Expr {
pos: p.tok.position() pos: p.tok.position()
is_used: true is_used: true
}) })
is_or_block_used = true or_kind = .block
or_stmts = p.parse_block_no_scope() or_stmts = p.parse_block_no_scope()
p.close_scope() p.close_scope()
} }
if p.tok.kind == .question {
// `foo()?`
p.next()
or_kind = .propagate
}
end_pos := p.tok.position() end_pos := p.tok.position()
pos := token.Position{ pos := token.Position{
line_nr: name_pos.line_nr line_nr: name_pos.line_nr
@ -1020,7 +1025,7 @@ fn (mut p Parser) dot_expr(left ast.Expr) ast.Expr {
is_method: true is_method: true
or_block: ast.OrExpr{ or_block: ast.OrExpr{
stmts: or_stmts stmts: or_stmts
is_used: is_or_block_used kind: or_kind
} }
} }
if is_filter { if is_filter {

View File

@ -93,6 +93,33 @@ fn foo_str() ?string {
return 'something' return 'something'
} }
fn propagate_optional(b bool) ?int {
a := err_call(b)?
return a
}
fn propagate_different_type(b bool) ?bool {
err_call(b)?
return true
}
fn test_propagation() {
a := propagate_optional(true) or {
0
}
assert a == 42
if _ := propagate_optional(false) {
assert false
}
b := propagate_different_type(true) or {
false
}
assert b == true
if _ := propagate_different_type(false) {
assert false
}
}
fn test_q() { fn test_q() {
// assert foo_ok()? == true // assert foo_ok()? == true
} }