checker: Fix array,map builtin methods can be called without lock (#13329)

pull/13338/head
Naoki MATSUMOTO 2022-02-01 21:02:00 +09:00
parent a014844050
commit 09bb0af59c
4 changed files with 38 additions and 3 deletions

View File

@ -1045,9 +1045,13 @@ pub fn (mut c Checker) method_call(mut node ast.CallExpr) ast.Type {
// TODO: remove this for actual methods, use only for compiler magic // TODO: remove this for actual methods, use only for compiler magic
// FIXME: Argument count != 1 will break these // FIXME: Argument count != 1 will break these
if left_sym.kind == .array && method_name in array_builtin_methods { if left_sym.kind == .array && method_name in array_builtin_methods {
// check if shared variable is locked
c.fail_if_unreadable(node.left, left_type, 'receiver')
return c.array_builtin_method_call(mut node, left_type, c.table.sym(left_type)) return c.array_builtin_method_call(mut node, left_type, c.table.sym(left_type))
} else if (left_sym.kind == .map || final_left_sym.kind == .map) } else if (left_sym.kind == .map || final_left_sym.kind == .map)
&& method_name in ['clone', 'keys', 'move', 'delete'] { && method_name in ['clone', 'keys', 'move', 'delete'] {
// check if shared variable is locked
c.fail_if_unreadable(node.left, left_type, 'receiver')
if left_sym.kind == .map { if left_sym.kind == .map {
return c.map_builtin_method_call(mut node, left_type, left_sym) return c.map_builtin_method_call(mut node, left_type, left_sym)
} else { } else {
@ -1055,6 +1059,8 @@ pub fn (mut c Checker) method_call(mut node ast.CallExpr) ast.Type {
return c.map_builtin_method_call(mut node, parent_type, final_left_sym) return c.map_builtin_method_call(mut node, parent_type, final_left_sym)
} }
} else if left_sym.kind == .array && method_name in ['insert', 'prepend'] { } else if left_sym.kind == .array && method_name in ['insert', 'prepend'] {
// check if shared variable is locked
c.fail_if_unreadable(node.left, left_type, 'receiver')
if method_name == 'insert' { if method_name == 'insert' {
if node.args.len != 2 { if node.args.len != 2 {
c.error('`array.insert()` should have 2 arguments, e.g. `insert(1, val)`', c.error('`array.insert()` should have 2 arguments, e.g. `insert(1, val)`',
@ -1083,6 +1089,8 @@ pub fn (mut c Checker) method_call(mut node ast.CallExpr) ast.Type {
c.error('cannot $method_name `$arg_sym.name` to `$left_sym.name`', arg_expr.pos()) c.error('cannot $method_name `$arg_sym.name` to `$left_sym.name`', arg_expr.pos())
} }
} else if final_left_sym.kind == .array && method_name in ['first', 'last', 'pop'] { } else if final_left_sym.kind == .array && method_name in ['first', 'last', 'pop'] {
// check if shared variable is locked
c.fail_if_unreadable(node.left, left_type, 'receiver')
if final_left_sym.info is ast.Array { if final_left_sym.info is ast.Array {
node.return_type = final_left_sym.info.elem_type node.return_type = final_left_sym.info.elem_type
return node.return_type return node.return_type
@ -1647,6 +1655,7 @@ fn (mut c Checker) map_builtin_method_call(mut node ast.CallExpr, left_type ast.
} else { } else {
ret_type = left_type ret_type = left_type
} }
ret_type = ret_type.clear_flag(.shared_f)
} }
'keys' { 'keys' {
info := left_sym.info as ast.Map info := left_sym.info as ast.Map
@ -1667,6 +1676,10 @@ fn (mut c Checker) map_builtin_method_call(mut node ast.CallExpr, left_type ast.
else {} else {}
} }
node.receiver_type = left_type.ref() node.receiver_type = left_type.ref()
// all methods does not take a lock inside them.
node.receiver_type = node.receiver_type.clear_flag(.shared_f)
node.return_type = ret_type node.return_type = ret_type
return node.return_type return node.return_type
} }
@ -1775,6 +1788,7 @@ fn (mut c Checker) array_builtin_method_call(mut node ast.CallExpr, left_type as
} else { } else {
node.return_type = node.receiver_type.set_nr_muls(0) node.return_type = node.receiver_type.set_nr_muls(0)
} }
node.return_type = node.return_type.clear_flag(.shared_f)
} else if method_name == 'sort' { } else if method_name == 'sort' {
node.return_type = ast.void_type node.return_type = ast.void_type
} else if method_name == 'contains' { } else if method_name == 'contains' {
@ -1791,6 +1805,10 @@ fn (mut c Checker) array_builtin_method_call(mut node ast.CallExpr, left_type as
node.receiver_type = left_type node.receiver_type = left_type
} }
} }
// all methods does not take a lock inside them.
node.receiver_type = node.receiver_type.clear_flag(.shared_f)
return node.return_type return node.return_type
} }

View File

@ -404,7 +404,9 @@ fn (mut g Gen) gen_array_sort(node ast.CallExpr) {
if node.args.len == 0 { if node.args.len == 0 {
comparison_type = g.unwrap(info.elem_type.set_nr_muls(0)) comparison_type = g.unwrap(info.elem_type.set_nr_muls(0))
shared a := g.array_sort_fn shared a := g.array_sort_fn
array_sort_fn := a.clone() array_sort_fn := rlock a {
a.clone()
}
if compare_fn in array_sort_fn { if compare_fn in array_sort_fn {
g.gen_array_sort_call(node, compare_fn) g.gen_array_sort_call(node, compare_fn)
return return
@ -425,7 +427,9 @@ fn (mut g Gen) gen_array_sort(node ast.CallExpr) {
compare_fn += '_reverse' compare_fn += '_reverse'
} }
shared a := g.array_sort_fn shared a := g.array_sort_fn
array_sort_fn := a.clone() array_sort_fn := rlock a {
a.clone()
}
if compare_fn in array_sort_fn { if compare_fn in array_sort_fn {
g.gen_array_sort_call(node, compare_fn) g.gen_array_sort_call(node, compare_fn)
return return

View File

@ -958,7 +958,9 @@ fn (mut g Gen) register_optional(t ast.Type) string {
} }
fn (mut g Gen) write_optionals() { fn (mut g Gen) write_optionals() {
mut done := g.done_optionals.clone() mut done := rlock g.done_optionals {
g.done_optionals.clone()
}
for base, styp in g.optionals { for base, styp in g.optionals {
if base in done { if base in done {
continue continue

View File

@ -121,3 +121,14 @@ fn test_shared_lock_chan_rec_expr() {
} }
} }
} }
fn test_shared_array_clone() {
shared a := []string{}
lock a {
a << "test"
}
b := rlock a {
a.clone()
}
assert b[0] == "test"
}