Skip to content

Commit 1aa3bbd

Browse files
committed
devirtualize v, ok := v.(T)
Change-Id: I6249fd63bc4734640055a2f764eb4e73a4e2adbb
1 parent 96ba2f9 commit 1aa3bbd

File tree

4 files changed

+46
-17
lines changed

4 files changed

+46
-17
lines changed

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

+12-4
Original file line numberDiff line numberDiff line change
@@ -892,21 +892,22 @@ func staticValue(n Node, forDevirt bool) Node {
892892
n = n1.X
893893
continue
894894
case *TypeAssertExpr:
895-
if forDevirt && n1.Op() == ODOTTYPE {
895+
if forDevirt {
896896
n = n1.X
897897
continue
898898
}
899899
}
900900

901-
n1 := staticValue1(n)
901+
n1 := staticValue1(n, forDevirt)
902902
if n1 == nil {
903903
return n
904904
}
905+
905906
n = n1
906907
}
907908
}
908909

909-
func staticValue1(nn Node) Node {
910+
func staticValue1(nn Node, forDevirt bool) Node {
910911
if nn.Op() != ONAME {
911912
return nil
912913
}
@@ -935,6 +936,14 @@ FindRHS:
935936
}
936937
}
937938
base.Fatalf("%v missing from LHS of %v", n, defn)
939+
case OAS2DOTTYPE:
940+
if !forDevirt {
941+
return nil
942+
}
943+
defn := defn.(*AssignListStmt)
944+
if defn.Lhs[0] == n {
945+
rhs = defn.Rhs[0]
946+
}
938947
default:
939948
return nil
940949
}
@@ -945,7 +954,6 @@ FindRHS:
945954
if Reassigned(n) {
946955
return nil
947956
}
948-
949957
return rhs
950958
}
951959

src/cmd/compile/internal/noder/reader.go

+2
Original file line numberDiff line numberDiff line change
@@ -2937,10 +2937,12 @@ func (r *reader) multiExpr() []ir.Node {
29372937
expr := r.expr()
29382938

29392939
results := make([]ir.Node, r.Len())
2940+
29402941
as := ir.NewAssignListStmt(pos, ir.OAS2, nil, []ir.Node{expr})
29412942
as.Def = true
29422943
for i := range results {
29432944
tmp := r.temp(pos, r.typ())
2945+
tmp.Defn = as
29442946
as.PtrInit().Append(ir.NewDecl(pos, ir.ODCL, tmp))
29452947
as.Lhs.Append(tmp)
29462948

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

+1
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,7 @@ func autotmpname(n int) string {
9292
s = fmt.Sprintf(prefix, n)
9393
autotmpnames[n] = s
9494
}
95+
9596
return s
9697
}
9798

test/escape_iface_with_devirt_type_assertions.go

+31-13
Original file line numberDiff line numberDiff line change
@@ -63,32 +63,31 @@ func t() {
6363
}
6464
}
6565

66-
// TODO: these type assertions could also be devirtualized.
6766
func t2() {
6867
{
69-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
68+
var a M = &Impl{} // ERROR "does not escape"
7069
if v, ok := a.(M); ok {
71-
v.M()
70+
v.M() // ERROR "devirtualizing" "inlining call"
7271
}
7372
}
7473
{
75-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
74+
var a M = &Impl{} // ERROR "does not escape"
7675
if v, ok := a.(A); ok {
77-
v.A()
76+
v.A() // ERROR "devirtualizing" "inlining call"
7877
}
7978
}
8079
{
81-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
80+
var a M = &Impl{} // ERROR "does not escape"
8281
v, ok := a.(M)
8382
if ok {
84-
v.M()
83+
v.M() // ERROR "devirtualizing" "inlining call"
8584
}
8685
}
8786
{
88-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
87+
var a M = &Impl{} // ERROR "does not escape"
8988
v, ok := a.(A)
9089
if ok {
91-
v.A()
90+
v.A() // ERROR "devirtualizing" "inlining call"
9291
}
9392
}
9493
{
@@ -100,21 +99,40 @@ func t2() {
10099
}
101100
}
102101
{
103-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
102+
var a M = &Impl{} // ERROR "does not escape"
104103
v, _ := a.(M)
105-
v.M()
104+
v.M() // ERROR "devirtualizing" "inlining call"
106105
}
107106
{
108-
var a M = &Impl{} // ERROR "&Impl{} escapes to heap"
107+
var a M = &Impl{} // ERROR "does not escape"
109108
v, _ := a.(A)
110-
v.A()
109+
v.A() // ERROR "devirtualizing" "inlining call"
111110
}
112111
{
113112
var a M = &Impl{} // ERROR "does not escape"
114113
v, _ := a.(*Impl)
115114
v.A() // ERROR "inlining"
116115
v.M() // ERROR "inlining"
117116
}
117+
{
118+
a := newM() // ERROR "does not escape" "inlining call"
119+
callA(a) // ERROR "devirtualizing" "inlining call"
120+
callIfA(a) // ERROR "devirtualizing" "inlining call"
121+
}
122+
}
123+
124+
func newM() M { // ERROR "can inline"
125+
return &Impl{} // ERROR "escapes"
126+
}
127+
128+
func callA(m M) { // ERROR "can inline" "leaking param"
129+
m.(A).A()
130+
}
131+
132+
func callIfA(m M) { // ERROR "can inline" "leaking param"
133+
if v, ok := m.(A); ok {
134+
v.A()
135+
}
118136
}
119137

120138
//go:noinline

0 commit comments

Comments
 (0)