Skip to content

Commit

Permalink
cmd/compile: use dictionary to convert type to shaped interface type
Browse files Browse the repository at this point in the history
When we convert a type to a shaped interface type, we are not able
to recognize the itab. So passing the itab by dictionary as the
workaround.

Fixes golang#52026.

Change-Id: I75c23c7dd215daf9761dc24116a8af2c28c6d948
Reviewed-on: https://go-review.googlesource.com/c/go/+/401034
Run-TryBot: Wayne Zuo <[email protected]>
TryBot-Result: Gopher Robot <[email protected]>
Auto-Submit: Keith Randall <[email protected]>
Reviewed-by: Keith Randall <[email protected]>
Reviewed-by: David Chase <[email protected]>
Reviewed-by: Keith Randall <[email protected]>
  • Loading branch information
wdvxdr1123 authored and gopherbot committed May 3, 2022
1 parent 23f1325 commit 64b6e44
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 4 deletions.
9 changes: 5 additions & 4 deletions src/cmd/compile/internal/noder/stencil.go
Original file line number Diff line number Diff line change
Expand Up @@ -1325,8 +1325,8 @@ func (g *genInst) dictPass(info *instInfo) {
mce := m.(*ir.ConvExpr)
// Note: x's argument is still typed as a type parameter.
// m's argument now has an instantiated type.
if mce.X.Type().HasShape() || (mce.X.Type().IsInterface() && m.Type().HasShape()) {
m = convertUsingDictionary(info, info.dictParam, m.Pos(), m.(*ir.ConvExpr).X, m, m.Type())
if mce.X.Type().HasShape() || m.Type().HasShape() {
m = convertUsingDictionary(info, info.dictParam, m.Pos(), mce.X, m, m.Type())
}
case ir.ODOTTYPE, ir.ODOTTYPE2:
if !m.Type().HasShape() {
Expand Down Expand Up @@ -1420,7 +1420,7 @@ func findDictType(info *instInfo, t *types.Type) int {
// instantiated node of the CONVIFACE node or XDOT node (for a bound method call) that is causing the
// conversion.
func convertUsingDictionary(info *instInfo, dictParam *ir.Name, pos src.XPos, v ir.Node, in ir.Node, dst *types.Type) ir.Node {
assert(v.Type().HasShape() || v.Type().IsInterface() && in.Type().HasShape())
assert(v.Type().HasShape() || in.Type().HasShape())
assert(dst.IsInterface())

if v.Type().IsInterface() {
Expand Down Expand Up @@ -1799,6 +1799,7 @@ func (g *genInst) finalizeSyms() {
g.instantiateMethods()
itabLsym := reflectdata.ITabLsym(srctype, dsttype)
d.off = objw.SymPtr(lsym, d.off, itabLsym, 0)
markTypeUsed(srctype, lsym)
infoPrint(" + Itab for (%v,%v)\n", srctype, dsttype)
}
}
Expand Down Expand Up @@ -1974,7 +1975,7 @@ func (g *genInst) getInstInfo(st *ir.Func, shapes []*types.Type, instInfo *instI
}
case ir.OCONVIFACE:
if n.Type().IsInterface() && !n.Type().IsEmptyInterface() &&
n.(*ir.ConvExpr).X.Type().HasShape() {
(n.Type().HasShape() || n.(*ir.ConvExpr).X.Type().HasShape()) {
infoPrint(" Itab for interface conv: %v\n", n)
info.itabConvs = append(info.itabConvs, n)
}
Expand Down
50 changes: 50 additions & 0 deletions test/typeparam/issue52026.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
// run

// Copyright 2022 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 returnOption[T any](n int) Option[T] {
if n == 1 {
return Some[T]{}
} else {
return None{}
}
}

type Option[T any] interface {
sealedOption()
}

type Some[T any] struct {
val T
}

func (s Some[T]) Value() T {
return s.val
}

func (s Some[T]) sealedOption() {}

type None struct{}

func (s None) sealedOption() {}

func main() {
s := returnOption[int](1)
_ = s.(Some[int])

s = returnOption[int](0)
_ = s.(None)

switch (any)(s).(type) {
case Some[int]:
panic("s is a Some[int]")
case None:
// ok
default:
panic("oops")
}
}

0 comments on commit 64b6e44

Please sign in to comment.