if isTyped(x.typ) && isTyped(y.typ) {
return false
}
+ // A numeric type can only convert to another numeric type.
+ if allNumeric(x.typ) != allNumeric(y.typ) {
+ return false
+ }
// An untyped operand may convert to its default type when paired with an empty interface
// TODO(gri) This should only matter for comparisons (the only binary operation that is
// valid with interfaces), but in that case the assignability check should take
if isTyped(x.typ) && isTyped(y.typ) {
return false
}
+ // A numeric type can only convert to another numeric type.
+ if allNumeric(x.typ) != allNumeric(y.typ) {
+ return false
+ }
// An untyped operand may convert to its default type when paired with an empty interface
// TODO(gri) This should only matter for comparisons (the only binary operation that is
// valid with interfaces), but in that case the assignability check should take
// context in which it is used.
//
// Example:
- // var _ = 1 + []int{}
+ // func f[T ~int8 | ~int16 | ~int32 | ~int64](x T) T {
+ // return x + 1024
+ // }
InvalidUntypedConversion
// BadOffsetofSyntax occurs when unsafe.Offsetof is called with an argument
var x2 P2
_ = max(x2)
_ = max(x2, x2)
- _ = max(1, 2 /* ERROR "cannot convert 2 (untyped int constant) to type P2" */, x2) // error at 2 because max is 2
+ _ = max(1, 2, x2 /* ERROR "mismatched types untyped int (previous argument) and P2 (type of x2)" */ )
_ = max(x1, x2 /* ERROR "mismatched types P1 (previous argument) and P2 (type of x2)" */ )
}
var x2 P2
_ = min(x2)
_ = min(x2, x2)
- _ = min(1 /* ERROR "cannot convert 1 (untyped int constant) to type P2" */ , 2, x2) // error at 1 because min is 1
+ _ = min(1, 2, x2 /* ERROR "mismatched types untyped int (previous argument) and P2 (type of x2)" */ )
_ = min(x1, x2 /* ERROR "mismatched types P1 (previous argument) and P2 (type of x2)" */ )
}
var s11 S11
var s2 S2
- _ = i == 0 /* ERROR "cannot convert" */
+ _ = i == 0 /* ERROR "invalid operation: i == 0 (mismatched types interface{m() int} and untyped int)" */
_ = i == s1 /* ERROR "mismatched types" */
_ = i == &s1
_ = i == &s11
func _() {
f(s.y /* ERROR "s.y undefined" */)
- f(1 /* ERROR "cannot convert 1" */ / s)
+ f(1 /* ERROR "invalid operation: 1 / s (mismatched types untyped int and struct{x int})" */ / s)
}
--- /dev/null
+// Copyright 2025 The Go Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+
+package p
+
+func f() {}
+
+const c = 0
+
+var v int
+var _ = f < c // ERROR "invalid operation: f < c (mismatched types func() and untyped int)"
+var _ = f < v // ERROR "invalid operation: f < v (mismatched types func() and int)"
var (
_ = b + 1 // ERROR "invalid operation.*mismatched types.*bool and untyped int"
_ = i + false // ERROR "invalid operation.*mismatched types.*int and untyped bool"
- _ = iface + 1 // ERROR "invalid operation.*mismatched types.*interface *{} and int"
- _ = iface + 1.0 // ERROR "invalid operation.*mismatched types.*interface *{} and float64"
+ _ = iface + 1 // ERROR "invalid operation.*mismatched types.*interface *{} and untyped int"
+ _ = iface + 1.0 // ERROR "invalid operation.*mismatched types.*interface *{} and untyped float"
_ = iface + false // ERROR "invalid operation.*mismatched types.*interface *{} and bool"
)