checker: fix generics with different order of generics fn (#9765)

pull/9777/head
yuyi 2021-04-17 13:29:06 +08:00 committed by GitHub
parent 990c4ab17a
commit 731e942055
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 24 additions and 22 deletions

View File

@ -6014,8 +6014,8 @@ pub fn (mut c Checker) map_init(mut node ast.MapInit) ast.Type {
info := c.table.get_type_symbol(node.typ).map_info() info := c.table.get_type_symbol(node.typ).map_info()
c.ensure_type_exists(info.key_type, node.pos) or {} c.ensure_type_exists(info.key_type, node.pos) or {}
c.ensure_type_exists(info.value_type, node.pos) or {} c.ensure_type_exists(info.value_type, node.pos) or {}
node.key_type = info.key_type node.key_type = c.unwrap_generic(info.key_type)
node.value_type = info.value_type node.value_type = c.unwrap_generic(info.value_type)
return node.typ return node.typ
} }
if node.keys.len > 0 && node.vals.len > 0 { if node.keys.len > 0 && node.vals.len > 0 {
@ -6052,13 +6052,12 @@ pub fn (mut c Checker) map_init(mut node ast.MapInit) ast.Type {
c.check_dup_keys(node, i) c.check_dup_keys(node, i)
} }
} }
key0_type = c.unwrap_generic(key0_type)
val0_type = c.unwrap_generic(val0_type)
mut map_type := ast.new_type(c.table.find_or_register_map(key0_type, val0_type)) mut map_type := ast.new_type(c.table.find_or_register_map(key0_type, val0_type))
node.typ = map_type node.typ = map_type
node.key_type = key0_type node.key_type = key0_type
node.value_type = val0_type node.value_type = val0_type
if node.key_type.has_flag(.generic) || node.value_type.has_flag(.generic) {
map_type = map_type.set_flag(.generic)
}
return map_type return map_type
} }
return node.typ return node.typ
@ -6319,6 +6318,8 @@ fn (mut c Checker) fetch_and_verify_orm_fields(info ast.Struct, pos token.Positi
fn (mut c Checker) post_process_generic_fns() { fn (mut c Checker) post_process_generic_fns() {
// Loop thru each generic function concrete type. // Loop thru each generic function concrete type.
// Check each specific fn instantiation. // Check each specific fn instantiation.
// Check 2 times (in order to check nested generics fn)
for _ in 0 .. 2 {
for i in 0 .. c.file.generic_fns.len { for i in 0 .. c.file.generic_fns.len {
if c.table.fn_generic_types.len == 0 { if c.table.fn_generic_types.len == 0 {
// no concrete types, so just skip: // no concrete types, so just skip:
@ -6335,6 +6336,7 @@ fn (mut c Checker) post_process_generic_fns() {
} }
node.cur_generic_types = [] node.cur_generic_types = []
} }
}
} }
fn (mut c Checker) fn_decl(mut node ast.FnDecl) { fn (mut c Checker) fn_decl(mut node ast.FnDecl) {

View File

@ -11,16 +11,16 @@ mut:
context Context context Context
} }
fn (ng NestedGeneric) nested_test<T>(mut app T) {
app.context = Context{}
}
fn method_test<T>(mut app T) int { fn method_test<T>(mut app T) int {
ng := NestedGeneric{} ng := NestedGeneric{}
ng.nested_test<T>(app) ng.nested_test<T>(app)
return 22 return 22
} }
fn (ng NestedGeneric) nested_test<T>(mut app T) {
app.context = Context{}
}
fn test_generics_with_generics_fn() { fn test_generics_with_generics_fn() {
mut app := App{} mut app := App{}
ret := method_test(mut app) ret := method_test(mut app)