all: interface arrays

pull/4645/head
Alexander Medvednikov 2020-04-29 12:20:22 +02:00
parent 3d8853af2b
commit 2fc05b814c
7 changed files with 103 additions and 65 deletions

View File

@ -644,6 +644,9 @@ pub:
has_cap bool has_cap bool
cap_expr Expr cap_expr Expr
mut: mut:
is_interface bool // array of interfaces e.g. `[]Animal` `[Dog{}, Cat{}]`
interface_types []table.Type // [Dog, Cat]
interface_type table.Type // Animal
elem_type table.Type elem_type table.Type
typ table.Type typ table.Type
} }

View File

@ -367,9 +367,8 @@ pub fn (mut c Checker) infix_expr(infix_expr mut ast.InfixExpr) table.Type {
return table.bool_type return table.bool_type
} }
.plus, .minus, .mul, .div { .plus, .minus, .mul, .div {
if infix_expr.op == .div && if infix_expr.op == .div && (infix_expr.right is ast.IntegerLiteral && infix_expr.right.str() ==
(infix_expr.right is ast.IntegerLiteral && infix_expr.right.str() == '0' || '0' || infix_expr.right is ast.FloatLiteral && infix_expr.right.str().f64() == 0.0) {
infix_expr.right is ast.FloatLiteral && infix_expr.right.str().f64() == 0.0) {
c.error('division by zero', infix_expr.right.position()) c.error('division by zero', infix_expr.right.position())
} }
if left.kind in [.array, .array_fixed, .map, .struct_] && !left.has_method(infix_expr.op.str()) { if left.kind in [.array, .array_fixed, .map, .struct_] && !left.has_method(infix_expr.op.str()) {
@ -743,7 +742,8 @@ pub fn (mut c Checker) call_fn(call_expr mut ast.CallExpr) table.Type {
if !found_in_args && call_expr.mod in ['builtin', 'main'] { if !found_in_args && call_expr.mod in ['builtin', 'main'] {
scope := c.file.scope.innermost(call_expr.pos.pos) scope := c.file.scope.innermost(call_expr.pos.pos)
if _ := scope.find_var(fn_name) { if _ := scope.find_var(fn_name) {
c.error('ambiguous call to: `$fn_name`, may refer to fn `$fn_name` or variable `$fn_name`', call_expr.pos) c.error('ambiguous call to: `$fn_name`, may refer to fn `$fn_name` or variable `$fn_name`',
call_expr.pos)
} }
} }
call_expr.return_type = f.return_type call_expr.return_type = f.return_type
@ -1117,27 +1117,49 @@ pub fn (mut c Checker) array_init(array_init mut ast.ArrayInit) table.Type {
} }
// [1,2,3] // [1,2,3]
if array_init.exprs.len > 0 && array_init.elem_type == table.void_type { if array_init.exprs.len > 0 && array_init.elem_type == table.void_type {
expecting_interface_array := c.expected_type != 0 && c.table.get_type_symbol(c.table.value_type(c.expected_type)).kind == mut expected_value_type := table.void_type
.interface_ mut expecting_interface_array := false
cap := array_init.exprs.len
mut interface_types := []table.Type{cap: cap}
if c.expected_type != 0 {
expected_value_type = c.table.value_type(c.expected_type)
if c.table.get_type_symbol(expected_value_type).kind == .interface_ {
// Array of interfaces? (`[dog, cat]`) Save the interface type (`Animal`)
expecting_interface_array = true
array_init.interface_type = expected_value_type
array_init.is_interface = true
}
}
// expecting_interface_array := c.expected_type != 0 &&
// c.table.get_type_symbol(c.table.value_type(c.expected_type)).kind == .interface_
//
// if expecting_interface_array { // if expecting_interface_array {
// println('ex $c.expected_type') // println('ex $c.expected_type')
// } // }
for i, expr in array_init.exprs { for i, expr in array_init.exprs {
typ := c.expr(expr) typ := c.expr(expr)
if expecting_interface_array {
if i == 0 {
elem_type = expected_value_type
c.expected_type = elem_type
}
interface_types << typ
continue
}
// The first element's type // The first element's type
if i == 0 { if i == 0 {
elem_type = typ elem_type = typ
c.expected_type = typ c.expected_type = typ
continue continue
} }
if expecting_interface_array {
continue
}
if !c.table.check(elem_type, typ) { if !c.table.check(elem_type, typ) {
elem_type_sym := c.table.get_type_symbol(elem_type) elem_type_sym := c.table.get_type_symbol(elem_type)
c.error('expected array element with type `$elem_type_sym.name`', array_init.pos) c.error('expected array element with type `$elem_type_sym.name`', array_init.pos)
} }
} }
if expecting_interface_array {
array_init.interface_types = interface_types
}
if array_init.is_fixed { if array_init.is_fixed {
idx := c.table.find_or_register_array_fixed(elem_type, array_init.exprs.len, 1) idx := c.table.find_or_register_array_fixed(elem_type, array_init.exprs.len, 1)
array_init.typ = table.new_type(idx) array_init.typ = table.new_type(idx)

View File

@ -996,7 +996,12 @@ fn (mut f Fmt) array_init(it ast.ArrayInit) {
if it.exprs.len == 0 && it.typ != 0 && it.typ != table.void_type { if it.exprs.len == 0 && it.typ != 0 && it.typ != table.void_type {
// `x := []string` // `x := []string`
f.write(f.type_to_str(it.typ)) f.write(f.type_to_str(it.typ))
f.write('{}') f.write('{')
if it.has_cap {
f.write('cap: ')
f.expr(it.cap_expr)
}
f.write('}')
return return
} }
// `[1,2,3]` // `[1,2,3]`

View File

@ -1176,7 +1176,7 @@ fn (mut g Gen) expr(node ast.Expr) {
g.write('.') g.write('.')
} }
if it.expr_type == 0 { if it.expr_type == 0 {
verror('cgen: SelectorExpr typ=0 field=$it.field') verror('cgen: SelectorExpr typ=0 field=$it.field $g.file.path $it.pos.line_nr')
} }
g.write(c_name(it.field)) g.write(c_name(it.field))
} }
@ -3361,7 +3361,17 @@ ${interface_name} I_${cctype}_to_${interface_name}(${cctype} x) {
fn (mut g Gen) array_init(it ast.ArrayInit) { fn (mut g Gen) array_init(it ast.ArrayInit) {
type_sym := g.table.get_type_symbol(it.typ) type_sym := g.table.get_type_symbol(it.typ)
if type_sym.kind != .array_fixed { if type_sym.kind == .array_fixed {
g.write('{')
for i, expr in it.exprs {
g.expr(expr)
if i != it.exprs.len - 1 {
g.write(', ')
}
}
g.write('}')
return
}
// elem_sym := g.table.get_type_symbol(it.elem_type) // elem_sym := g.table.get_type_symbol(it.elem_type)
elem_type_str := g.typ(it.elem_type) elem_type_str := g.typ(it.elem_type)
if it.exprs.len == 0 { if it.exprs.len == 0 {
@ -3379,24 +3389,22 @@ fn (mut g Gen) array_init(it ast.ArrayInit) {
g.write('0, ') g.write('0, ')
} }
g.write('sizeof($elem_type_str))') g.write('sizeof($elem_type_str))')
} else { return
}
len := it.exprs.len len := it.exprs.len
g.write('new_array_from_c_array($len, $len, sizeof($elem_type_str), ') g.write('new_array_from_c_array($len, $len, sizeof($elem_type_str), ')
g.write('($elem_type_str[$len]){\n\t\t') g.write('($elem_type_str[$len]){\n\t\t')
for expr in it.exprs { for i, expr in it.exprs {
if it.is_interface {
sym := g.table.get_type_symbol(it.interface_types[i])
isym := g.table.get_type_symbol(it.interface_type)
g.write('I_${sym.name}_to_${isym.name}(')
}
g.expr(expr) g.expr(expr)
if it.is_interface {
g.write(')')
}
g.write(', ') g.write(', ')
} }
g.write('\n})') g.write('\n})')
} }
} else {
g.write('{')
for i, expr in it.exprs {
g.expr(expr)
if i != it.exprs.len - 1 {
g.write(', ')
}
}
g.write('}')
}
}

View File

@ -201,9 +201,9 @@ pub fn (mut p Parser) close_scope() {
ast.Var { ast.Var {
if !it.is_used && !it.name.starts_with('__') { if !it.is_used && !it.name.starts_with('__') {
if p.pref.is_prod { if p.pref.is_prod {
p.error_with_pos('Unused variable: $it.name', it.pos) p.error_with_pos('unused variable: `$it.name`', it.pos)
} else { } else {
p.warn_with_pos('Unused variable: $it.name', it.pos) p.warn_with_pos('unused variable: `$it.name`', it.pos)
} }
} }
} }
@ -1249,7 +1249,6 @@ fn (mut p Parser) assoc() ast.Assoc {
fields: fields fields: fields
exprs: vals exprs: vals
pos: pos pos: pos
// typ: v.typ
} }
} }

View File

@ -201,7 +201,7 @@ pub fn (t &Table) get_type_symbol(typ Type) &TypeSymbol {
return &t.types[idx] return &t.types[idx]
} }
// this should never happen // this should never happen
panic('get_type_symbol: invalid type (typ=$typ idx=${idx}). This should never happen') panic('get_type_symbol: invalid type (typ=$typ idx=${idx}). Compiler bug. This should never happen')
} }
[inline] [inline]

View File

@ -46,7 +46,7 @@ fn test_perform_speak() {
perform_speak(dog) perform_speak(dog)
cat := Cat{} cat := Cat{}
perform_speak(cat) perform_speak(cat)
//perform_speakers([dog, cat]) perform_speakers([dog, cat])
/* /*
f := Foo { f := Foo {
speaker: dog speaker: dog
@ -62,6 +62,7 @@ interface Speak2er {
speak() speak()
} }
struct Foo { struct Foo {
speaker Speaker speaker Speaker
speakers []Speaker speakers []Speaker