// and thus the hash is irrelevant.
// Note: m.Hash is _not_ the hash used for the runtime itabTable hash table.
m.Hash = 0
- itabInit(m)
+ itabInit(m, true)
itabAdd(m)
unlock(&itabLock)
finish:
// The cached result doesn't record which
// interface function was missing, so initialize
// the itab again to get the missing function name.
- panic(&TypeAssertionError{concrete: typ, asserted: &inter.Type, missingMethod: itabInit(m)})
+ panic(&TypeAssertionError{concrete: typ, asserted: &inter.Type, missingMethod: itabInit(m, false)})
}
// find finds the given interface/type pair in t.
}
}
-// init fills in the m.Fun array with all the code pointers for
+// itabInit fills in the m.Fun array with all the code pointers for
// the m.Inter/m.Type pair. If the type does not implement the interface,
// it sets m.Fun[0] to 0 and returns the name of an interface function that is missing.
-// It is ok to call this multiple times on the same m, even concurrently.
-func itabInit(m *itab) string {
+// If !firstTime, itabInit will not write anything to m.Fun (see issue 65962).
+// It is ok to call this multiple times on the same m, even concurrently
+// (although it will only be called once with firstTime==true).
+func itabInit(m *itab, firstTime bool) string {
inter := m.Inter
typ := m.Type
x := typ.Uncommon()
ifn := rtyp.textOff(t.Ifn)
if k == 0 {
fun0 = ifn // we'll set m.Fun[0] at the end
- } else {
+ } else if firstTime {
methods[k] = ifn
}
continue imethods
}
}
// didn't find method
- m.Fun[0] = 0
+ // Leaves m.Fun[0] set to 0.
return iname
}
- m.Fun[0] = uintptr(fun0)
+ if firstTime {
+ m.Fun[0] = uintptr(fun0)
+ }
return ""
}
--- /dev/null
+// run
+
+// Copyright 2024 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 main
+
+func main() {
+ test1()
+ test2()
+}
+
+type I interface {
+ f()
+ g()
+ h()
+}
+
+//go:noinline
+func ld[T any]() {
+ var x I
+ if _, ok := x.(T); ok {
+ }
+}
+
+func isI(x any) {
+ _ = x.(I)
+}
+
+func test1() {
+ defer func() { recover() }()
+ ld[bool]() // add <bool,I> itab to binary
+ _ = any(false).(I)
+}
+
+type B bool
+
+func (B) f() {
+}
+func (B) g() {
+}
+
+func test2() {
+ defer func() { recover() }()
+ ld[B]() // add <B,I> itab to binary
+ _ = any(B(false)).(I)
+}