checker: refactor and clean up c.check_basic() (#8508)

pull/8534/head
Ruofan XU 2021-02-03 17:57:06 +08:00 committed by GitHub
parent 4b99d6af95
commit a73c20916d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 22 additions and 80 deletions

View File

@ -22,110 +22,52 @@ pub fn (mut c Checker) check_expected_call_arg(got table.Type, expected_ table.T
} }
pub fn (mut c Checker) check_basic(got table.Type, expected table.Type) bool { pub fn (mut c Checker) check_basic(got table.Type, expected table.Type) bool {
if got == expected { got_, exp_ := c.table.unalias_num_type(got), c.table.unalias_num_type(expected)
return true if got_.idx() == exp_.idx() {
}
t := c.table
got_idx := t.unalias_num_type(got).idx()
exp_idx := t.unalias_num_type(expected).idx()
// exp_is_optional := expected.has_flag(.optional)
// got_is_optional := got.has_flag(.optional)
// if (exp_is_optional && !got_is_optional) || (!exp_is_optional && got_is_optional) {
// return false
//}
// println('check: $got_type_sym.name, $exp_type_sym.name')
// # NOTE: use idxs here, and symbols below for perf
// got_is_ptr := got.is_ptr()
if got_idx == exp_idx {
// this is returning true even if one type is a ptr // this is returning true even if one type is a ptr
// and the other is not, is this correct behaviour? // and the other is not, is this correct behaviour?
return true return true
} }
if got_idx == table.none_type_idx && expected.has_flag(.optional) { if (exp_.is_pointer() || exp_.is_number()) && (got_.is_pointer() || got_.is_number()) {
return false return true
} }
exp_is_ptr := expected.is_ptr()
// allow pointers to be initialized with 0. TODO: use none instead // allow pointers to be initialized with 0. TODO: use none instead
if exp_is_ptr && got_idx == table.int_type_idx { if expected.is_ptr() && got_.idx() == table.int_type_idx {
return true return true
} }
if exp_idx == table.voidptr_type_idx || got_idx == table.voidptr_type_idx { // TODO: use sym so it can be absorbed into below [.voidptr, .any] logic
if expected.idx() == table.array_type_idx || got.idx() == table.array_type_idx {
return true return true
} }
if exp_idx == table.any_type_idx || got_idx == table.any_type_idx { got_sym, exp_sym := c.table.get_type_symbol(got), c.table.get_type_symbol(expected)
return true // array/map as argument
} if got_sym.kind in [.array, .map, .array_fixed] && exp_sym.kind == got_sym.kind {
// TODO i64 as int etc
if (exp_idx in table.pointer_type_idxs || exp_idx in table.number_type_idxs)
&& (got_idx in table.pointer_type_idxs || got_idx in table.number_type_idxs) {
return true
}
// if exp_idx in pointer_type_idxs && got_idx in pointer_type_idxs {
// return true
// }
// see hack in checker IndexExpr line #691
if (got_idx == table.byte_type_idx && exp_idx == table.byteptr_type_idx)
|| (exp_idx == table.byte_type_idx && got_idx == table.byteptr_type_idx) {
return true
}
if (got_idx == table.char_type_idx && exp_idx == table.charptr_type_idx)
|| (exp_idx == table.char_type_idx && got_idx == table.charptr_type_idx) {
return true
}
// TODO: this should no longer be needed
// if expected == table.t_type && got == table.t_type {
// return true
// }
// # NOTE: use symbols from this point on for perf
got_type_sym := t.get_type_symbol(got)
exp_type_sym := t.get_type_symbol(expected)
//
if exp_type_sym.kind == .function && got_type_sym.kind == .int {
// TODO temporary
// fn == 0
return true
}
// array/map fn
if got_type_sym.kind in [.array, .map] && exp_type_sym.kind == got_type_sym.kind {
if c.table.type_to_str(got) == c.table.type_to_str(expected).trim('&') { if c.table.type_to_str(got) == c.table.type_to_str(expected).trim('&') {
return true return true
} }
} }
// fixed array fn // e.g. [4096]byte vs byteptr
if got_type_sym.kind == .array_fixed && exp_type_sym.kind == .array_fixed { if got_sym.kind == .array_fixed {
if c.table.type_to_str(got) == c.table.type_to_str(expected).trim('&') { info := got_sym.info as table.ArrayFixed
if c.table.type_to_str(info.elem_type) == c.table.type_to_str(expected).trim('ptr') {
return true return true
} }
} }
if got_type_sym.kind == .array_fixed && exp_type_sym.kind == .byteptr { if exp_sym.kind in [.voidptr, .any] || got_sym.kind in [.voidptr, .any] {
info := got_type_sym.info as table.ArrayFixed
if info.elem_type.idx() == table.byte_type_idx {
return true
}
}
// TODO
// if exp_type_sym.name == 'array' || got_type_sym.name == 'array' {
if got_idx == table.array_type_idx || exp_idx == table.array_type_idx {
return true
}
// TODO
// accept [] when an expected type is an array
if got_type_sym.kind == .array && exp_type_sym.kind == .array
&& got_type_sym.name == 'array_void' {
return true
}
// type alias
if (got_type_sym.kind == .alias && got_type_sym.parent_idx == exp_idx)
|| (exp_type_sym.kind == .alias && exp_type_sym.parent_idx == got_idx) {
return true return true
} }
// sum type // sum type
if c.table.sumtype_has_variant(expected, c.table.mktyp(got)) { if c.table.sumtype_has_variant(expected, c.table.mktyp(got)) {
return true return true
} }
// type alias
if (got_sym.kind == .alias && got_sym.parent_idx == expected.idx())
|| (exp_sym.kind == .alias && exp_sym.parent_idx == got.idx()) {
return true
}
// fn type // fn type
if got_type_sym.kind == .function && exp_type_sym.kind == .function { if got_sym.kind == .function && exp_sym.kind == .function {
return c.check_matching_function_symbols(got_type_sym, exp_type_sym) return c.check_matching_function_symbols(got_sym, exp_sym)
} }
return false return false
} }