checker, cgen: fix generic operator overload (fix #11472) (#11479)

pull/11489/head
yuyi 2021-09-13 14:49:28 +08:00 committed by GitHub
parent 012da10517
commit 30029eaf5d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 71 additions and 18 deletions

View File

@ -1434,6 +1434,10 @@ pub fn (mut c Checker) infix_expr(mut node ast.InfixExpr) ast.Type {
} else { } else {
return_type = left_type return_type = left_type
} }
} else {
if left_sym.kind == .struct_
&& (left_sym.info as ast.Struct).generic_types.len > 0 {
return_type = left_type
} else { } else {
left_name := c.table.type_to_str(left_type) left_name := c.table.type_to_str(left_type)
right_name := c.table.type_to_str(right_type) right_name := c.table.type_to_str(right_type)
@ -1441,7 +1445,9 @@ pub fn (mut c Checker) infix_expr(mut node ast.InfixExpr) ast.Type {
c.error('undefined operation `$left_name` $node.op.str() `$right_name`', c.error('undefined operation `$left_name` $node.op.str() `$right_name`',
left_right_pos) left_right_pos)
} else { } else {
c.error('mismatched types `$left_name` and `$right_name`', left_right_pos) c.error('mismatched types `$left_name` and `$right_name`',
left_right_pos)
}
} }
} }
} else if right_sym.kind in [.array, .array_fixed, .map, .struct_] { } else if right_sym.kind in [.array, .array_fixed, .map, .struct_] {

View File

@ -468,6 +468,17 @@ fn (mut g Gen) gen_interface_is_op(node ast.InfixExpr) {
fn (mut g Gen) infix_expr_arithmetic_op(node ast.InfixExpr) { fn (mut g Gen) infix_expr_arithmetic_op(node ast.InfixExpr) {
left := g.unwrap(node.left_type) left := g.unwrap(node.left_type)
right := g.unwrap(node.right_type) right := g.unwrap(node.right_type)
if left.sym.kind == .struct_ && (left.sym.info as ast.Struct).generic_types.len > 0 {
concrete_types := (left.sym.info as ast.Struct).concrete_types
mut method_name := left.sym.cname + '_' + util.replace_op(node.op.str())
method_name = g.generic_fn_name(concrete_types, method_name, true)
g.write(method_name)
g.write('(')
g.expr(node.left)
g.write(', ')
g.expr(node.right)
g.write(')')
} else {
method := g.table.type_find_method(left.sym, node.op.str()) or { method := g.table.type_find_method(left.sym, node.op.str()) or {
g.gen_plain_infix_expr(node) g.gen_plain_infix_expr(node)
return return
@ -482,6 +493,7 @@ fn (mut g Gen) infix_expr_arithmetic_op(node ast.InfixExpr) {
g.op_arg(node.right, method.params[1].typ, right.typ) g.op_arg(node.right, method.params[1].typ, right.typ)
g.write(')') g.write(')')
} }
}
// infix_expr_left_shift_op generates code for the `<<` operator // infix_expr_left_shift_op generates code for the `<<` operator
// This can either be a value pushed into an array or a bit shift // This can either be a value pushed into an array or a bit shift

View File

@ -0,0 +1,35 @@
struct Matrix<T> {
row int
col int
mut:
data [][]T
}
fn from_array<T>(arr [][]T) Matrix<T> {
return Matrix<T>{
row: arr.len
col: arr[0].len
data: arr.clone()
}
}
fn (m1 Matrix<T>) + (m2 Matrix<T>) Matrix<T> {
if m1.row != m2.row || m1.col != m2.col {
panic('Addition can only be performed on matrix with same size')
}
mut res := m1
for i in 0 .. m2.row {
for j in 0 .. m2.col {
res.data[i][j] += m2.data[i][j]
}
}
return res
}
fn test_generic_operator_overload() {
result := from_array([[1, 2, 3], [4, 5, 6]]) + from_array([[7, 8, 9], [10, 11, 12]])
println(result)
assert result.row == 2
assert result.col == 3
assert result.data == [[8, 10, 12], [14, 16, 18]]
}