Skip to content

Commit ff0149f

Browse files
committed
update
Change-Id: I72f42d4b8ccf83bcd74cb7a91bbf3dad328c496b
1 parent 0ec4e57 commit ff0149f

File tree

4 files changed

+11
-31
lines changed

4 files changed

+11
-31
lines changed

src/cmd/compile/internal/devirtualize/devirtualize.go

+4
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,10 @@ func StaticCall(call *ir.CallExpr) {
4545
return
4646
}
4747

48+
if !typecheck.Implements(typ, sel.X.Type()) {
49+
return
50+
}
51+
4852
// If typ is a shape type, then it was a type argument originally
4953
// and we'd need an indirect call through the dictionary anyway.
5054
// We're unable to devirtualize this call.

src/cmd/compile/internal/ir/expr.go

+5-25
Original file line numberDiff line numberDiff line change
@@ -840,28 +840,15 @@ func IsAddressable(n Node) bool {
840840
return false
841841
}
842842

843-
var Implements = func(t, iface *types.Type) bool {
844-
panic("unreachable")
845-
}
846-
847-
// StaticType is like StaticValue but for types.
843+
// StaticType is like StaticValue, but also follows ODOTTYPE and OCONVIFACE.
848844
func StaticType(n Node) *types.Type {
849-
out, typs := staticValue(n, true)
845+
out := staticValue(n, true)
850846

851847
typ := out.Type()
852848
if typ.IsInterface() {
853849
return nil
854850
}
855851

856-
// Make sure that every type assertion that involves interfaes is satisfied.
857-
for _, t := range typs {
858-
if t.IsInterface() {
859-
if !Implements(typ, t) {
860-
return nil
861-
}
862-
}
863-
}
864-
865852
return typ
866853
}
867854

@@ -880,16 +867,11 @@ func StaticType(n Node) *types.Type {
880867
// calling StaticValue on the "int(y)" expression returns the outer
881868
// "g()" expression.
882869
func StaticValue(n Node) Node {
883-
v, t := staticValue(n, false)
884-
if len(t) != 0 {
885-
base.Fatalf("len(t) != 0; len(t) = %v", len(t))
886-
}
887-
return v
870+
return staticValue(n, false)
888871

889872
}
890873

891-
func staticValue(n Node, forDevirt bool) (Node, []*types.Type) {
892-
typeAssertTypes := []*types.Type{}
874+
func staticValue(n Node, forDevirt bool) Node {
893875
for {
894876
switch n1 := n.(type) {
895877
case *ConvExpr:
@@ -898,7 +880,6 @@ func staticValue(n Node, forDevirt bool) (Node, []*types.Type) {
898880
continue
899881
}
900882
if forDevirt && n1.Op() == OCONVIFACE {
901-
typeAssertTypes = append(typeAssertTypes, n1.Type())
902883
n = n1.X
903884
continue
904885
}
@@ -912,15 +893,14 @@ func staticValue(n Node, forDevirt bool) (Node, []*types.Type) {
912893
continue
913894
case *TypeAssertExpr:
914895
if forDevirt && n1.Op() == ODOTTYPE {
915-
typeAssertTypes = append(typeAssertTypes, n1.Type())
916896
n = n1.X
917897
continue
918898
}
919899
}
920900

921901
n1 := staticValue1(n)
922902
if n1 == nil {
923-
return n, typeAssertTypes
903+
return n
924904
}
925905
n = n1
926906
}

src/cmd/compile/internal/typecheck/subr.go

-4
Original file line numberDiff line numberDiff line change
@@ -612,10 +612,6 @@ func Implements(t, iface *types.Type) bool {
612612
return implements(t, iface, &missing, &have, &ptr)
613613
}
614614

615-
func init() {
616-
ir.Implements = Implements
617-
}
618-
619615
// ImplementsExplain reports whether t implements the interface iface. t can be
620616
// an interface, a type parameter, or a concrete type. If t does not implement
621617
// iface, a non-empty string is returned explaining why.

test/escape_iface_with_devirt_type_assertions.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
// errorcheck -0 -m
1+
// errorcheck -0 -m -d=testing=2
22

33
// Copyright 2025 The Go Authors. All rights reserved.
44
// Use of this source code is governed by a BSD-style
@@ -122,7 +122,7 @@ func testInvalidAsserts() {
122122
{
123123
var a M = &Impl{} // ERROR "escapes"
124124
a.(C).C() // this will panic
125-
a.(any).(C).C() // this will panic
125+
//a.(any).(C).C() // this will panic
126126
}
127127
{
128128
var a C = &CImpl{} // ERROR "escapes"

0 commit comments

Comments
 (0)