checker,cgen: fix if expressions in lock expression (#14384)

master
crthpl 2022-05-15 02:31:07 -07:00 committed by GitHub
parent c28051020a
commit e4065bd57b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 37 additions and 5 deletions

View File

@ -3537,6 +3537,7 @@ pub fn (mut c Checker) select_expr(mut node ast.SelectExpr) ast.Type {
} }
pub fn (mut c Checker) lock_expr(mut node ast.LockExpr) ast.Type { pub fn (mut c Checker) lock_expr(mut node ast.LockExpr) ast.Type {
expected_type := c.expected_type
if c.rlocked_names.len > 0 || c.locked_names.len > 0 { if c.rlocked_names.len > 0 || c.locked_names.len > 0 {
c.error('nested `lock`/`rlock` not allowed', node.pos) c.error('nested `lock`/`rlock` not allowed', node.pos)
} }
@ -3560,16 +3561,17 @@ pub fn (mut c Checker) lock_expr(mut node ast.LockExpr) ast.Type {
} }
} }
c.stmts(node.stmts) c.stmts(node.stmts)
c.rlocked_names = []
c.locked_names = []
// handle `x := rlock a { a.getval() }` // handle `x := rlock a { a.getval() }`
mut ret_type := ast.void_type mut ret_type := ast.void_type
if node.stmts.len > 0 { if node.stmts.len > 0 {
last_stmt := node.stmts.last() last_stmt := node.stmts.last()
if last_stmt is ast.ExprStmt { if last_stmt is ast.ExprStmt {
ret_type = last_stmt.typ c.expected_type = expected_type
ret_type = c.expr(last_stmt.expr)
} }
} }
c.rlocked_names = []
c.locked_names = []
if ret_type != ast.void_type { if ret_type != ast.void_type {
node.is_expr = true node.is_expr = true
} }

View File

@ -5,6 +5,13 @@ vlib/v/checker/tests/lock_already_locked.vv:11:3: error: nested `lock`/`rlock` n
| ~~~~~ | ~~~~~
12 | a.x++ 12 | a.x++
13 | } 13 | }
vlib/v/checker/tests/lock_already_locked.vv:12:4: error: a has an `rlock` but needs a `lock`
10 | lock a {
11 | rlock a {
12 | a.x++
| ^
13 | }
14 | }
vlib/v/checker/tests/lock_already_locked.vv:15:10: error: `a` is `shared` and must be `rlock`ed or `lock`ed to be used as non-mut argument to print vlib/v/checker/tests/lock_already_locked.vv:15:10: error: `a` is `shared` and must be `rlock`ed or `lock`ed to be used as non-mut argument to print
13 | } 13 | }
14 | } 14 | }

View File

@ -456,6 +456,8 @@ fn (mut g Gen) gen_assign_stmt(node_ ast.AssignStmt) {
} }
if val is ast.ArrayInit { if val is ast.ArrayInit {
g.array_init(val, ident.name) g.array_init(val, ident.name)
} else if val_type.has_flag(.shared_f) {
g.expr_with_cast(val, val_type, var_type)
} else { } else {
g.expr(val) g.expr(val)
} }

View File

@ -78,7 +78,8 @@ fn (mut g Gen) if_expr(node ast.IfExpr) {
g.write(' ? ') g.write(' ? ')
} }
prev_expected_cast_type := g.expected_cast_type prev_expected_cast_type := g.expected_cast_type
if node.is_expr && g.table.sym(node.typ).kind == .sum_type { if node.is_expr
&& (g.table.sym(node.typ).kind == .sum_type || node.typ.has_flag(.shared_f)) {
g.expected_cast_type = node.typ g.expected_cast_type = node.typ
} }
g.stmts(branch.stmts) g.stmts(branch.stmts)
@ -204,7 +205,8 @@ fn (mut g Gen) if_expr(node ast.IfExpr) {
} }
if needs_tmp_var { if needs_tmp_var {
prev_expected_cast_type := g.expected_cast_type prev_expected_cast_type := g.expected_cast_type
if node.is_expr && g.table.sym(node.typ).kind == .sum_type { if node.is_expr
&& (g.table.sym(node.typ).kind == .sum_type || node.typ.has_flag(.shared_f)) {
g.expected_cast_type = node.typ g.expected_cast_type = node.typ
} }
g.stmts_with_tmp_var(branch.stmts, tmp) g.stmts_with_tmp_var(branch.stmts, tmp)

View File

@ -0,0 +1,19 @@
type AA = bool | int
fn test_shared_if_expr() {
shared a := [1, 2, 3]
b := [4, 5, 6]
c := lock a {
if a == b { a } else { b }
}
assert c == [4, 5, 6]
d := lock a {
if a != b {
a << 5
a
} else {
b
}
}
assert d == [1, 2, 3, 5]
}