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

checker: fix generics fn inferring map type (fix #10835) (#10839)

This commit is contained in:
yuyi 2021-07-17 22:49:21 +08:00 committed by GitHub
parent 6bf69ea3e3
commit 3a0fa550b6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 48 additions and 1 deletions

View File

@ -667,6 +667,17 @@ pub fn (mut c Checker) infer_fn_generic_types(f ast.Fn, mut call_expr ast.CallEx
break break
} }
} }
} else if arg_sym.kind == .map && param_type_sym.kind == .map {
arg_map_info := arg_sym.info as ast.Map
param_map_info := param_type_sym.info as ast.Map
if param_map_info.key_type.has_flag(.generic)
&& c.table.get_type_symbol(param_map_info.key_type).name == gt_name {
typ = arg_map_info.key_type
}
if param_map_info.value_type.has_flag(.generic)
&& c.table.get_type_symbol(param_map_info.value_type).name == gt_name {
typ = arg_map_info.value_type
}
} else if param.typ.has_flag(.variadic) { } else if param.typ.has_flag(.variadic) {
to_set = c.table.mktyp(arg.typ) to_set = c.table.mktyp(arg.typ)
} else if arg_sym.kind in [.struct_, .interface_, .sum_type] { } else if arg_sym.kind in [.struct_, .interface_, .sum_type] {

View File

@ -99,7 +99,7 @@ pub fn (mut p Parser) parse_map_type() ast.Type {
return 0 return 0
} }
key_type_supported := key_type in [ast.string_type_idx, ast.voidptr_type_idx] key_type_supported := key_type in [ast.string_type_idx, ast.voidptr_type_idx]
|| key_sym.kind == .enum_ || key_sym.kind == .placeholder || key_sym.kind in [.enum_, .placeholder, .any]
|| ((key_type.is_int() || key_type.is_float() || is_alias) && !key_type.is_ptr()) || ((key_type.is_int() || key_type.is_float() || is_alias) && !key_type.is_ptr())
if !key_type_supported { if !key_type_supported {
if is_alias { if is_alias {

View File

@ -0,0 +1,36 @@
fn print_map<K, V>(x map[K]V) string {
println(x)
return '$x'
}
fn test_generics_infer_map_type() {
m1 := map{
'one': 1
}
assert print_map(m1) == "{'one': 1}"
m2 := map{
'one': 1.1
}
assert print_map(m2) == "{'one': 1.1}"
m3 := map{
'one': 'a'
}
assert print_map(m3) == "{'one': 'a'}"
m4 := map{
1: 'one'
}
assert print_map(m4) == "{1: 'one'}"
m5 := map{
1.1: 'one'
}
assert print_map(m5) == "{1.1: 'one'}"
m6 := map{
'a': 'one'
}
assert print_map(m6) == "{'a': 'one'}"
}