mirror of
https://github.com/golang/go.git
synced 2025-10-23 21:13:20 +00:00
[dev.typeparams] cmd/compile: handle calling a method on a type param in stenciling
- Have to delay the extra transformation on methods invoked on a type param, since the actual transformation (including path through embedded fields) will depend on the instantiated type. I am currently doing the transformation during the stencil substitution phase. We probably should have a separate pass after noder2 and stenciling, which drives the extra transformations that were in the old typechecker. - We handle method values (that are not called) and method calls. We don't currently handle method expressions. - Handle type substitution in function types, which is needed for function args in generic functions. - Added stringer.go and map.go tests, testing the above changes (including constraints with embedded interfaces). Change-Id: I3831a937d2b8814150f75bebf9f23ab10b93fa00 Reviewed-on: https://go-review.googlesource.com/c/go/+/290550 TryBot-Result: Go Bot <gobot@golang.org> Trust: Dan Scales <danscales@google.com> Trust: Robert Griesemer <gri@golang.org> Run-TryBot: Dan Scales <danscales@google.com> Reviewed-by: Robert Griesemer <gri@golang.org>
This commit is contained in:
parent
ca18c42054
commit
12e15d430d
6 changed files with 220 additions and 16 deletions
|
|
@ -87,11 +87,13 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
|
|||
|
||||
case *syntax.CompositeLit:
|
||||
return g.compLit(typ, expr)
|
||||
|
||||
case *syntax.FuncLit:
|
||||
return g.funcLit(typ, expr)
|
||||
|
||||
case *syntax.AssertExpr:
|
||||
return Assert(pos, g.expr(expr.X), g.typeExpr(expr.Type))
|
||||
|
||||
case *syntax.CallExpr:
|
||||
fun := g.expr(expr.Fun)
|
||||
if inferred, ok := g.info.Inferred[expr]; ok && len(inferred.Targs) > 0 {
|
||||
|
|
@ -114,6 +116,7 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
|
|||
|
||||
}
|
||||
return Call(pos, g.typ(typ), fun, g.exprs(expr.ArgList), expr.HasDots)
|
||||
|
||||
case *syntax.IndexExpr:
|
||||
var targs []ir.Node
|
||||
if _, ok := expr.Index.(*syntax.ListExpr); ok {
|
||||
|
|
@ -139,6 +142,7 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
|
|||
|
||||
case *syntax.ParenExpr:
|
||||
return g.expr(expr.X) // skip parens; unneeded after parse+typecheck
|
||||
|
||||
case *syntax.SelectorExpr:
|
||||
// Qualified identifier.
|
||||
if name, ok := expr.X.(*syntax.Name); ok {
|
||||
|
|
@ -147,8 +151,8 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
|
|||
return typecheck.Expr(g.use(expr.Sel))
|
||||
}
|
||||
}
|
||||
return g.selectorExpr(pos, typ, expr)
|
||||
|
||||
return g.selectorExpr(pos, expr)
|
||||
case *syntax.SliceExpr:
|
||||
return Slice(pos, g.expr(expr.X), g.expr(expr.Index[0]), g.expr(expr.Index[1]), g.expr(expr.Index[2]))
|
||||
|
||||
|
|
@ -172,15 +176,22 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
|
|||
// selectorExpr resolves the choice of ODOT, ODOTPTR, OCALLPART (eventually
|
||||
// ODOTMETH & ODOTINTER), and OMETHEXPR and deals with embedded fields here rather
|
||||
// than in typecheck.go.
|
||||
func (g *irgen) selectorExpr(pos src.XPos, expr *syntax.SelectorExpr) ir.Node {
|
||||
selinfo := g.info.Selections[expr]
|
||||
func (g *irgen) selectorExpr(pos src.XPos, typ types2.Type, expr *syntax.SelectorExpr) ir.Node {
|
||||
x := g.expr(expr.X)
|
||||
if x.Type().Kind() == types.TTYPEPARAM {
|
||||
// Leave a method call on a type param as an OXDOT, since it can
|
||||
// only be fully transformed once it has an instantiated type.
|
||||
n := ir.NewSelectorExpr(pos, ir.OXDOT, x, typecheck.Lookup(expr.Sel.Value))
|
||||
typed(g.typ(typ), n)
|
||||
return n
|
||||
}
|
||||
|
||||
selinfo := g.info.Selections[expr]
|
||||
// Everything up to the last selection is an implicit embedded field access,
|
||||
// and the last selection is determined by selinfo.Kind().
|
||||
index := selinfo.Index()
|
||||
embeds, last := index[:len(index)-1], index[len(index)-1]
|
||||
|
||||
x := g.expr(expr.X)
|
||||
origx := x
|
||||
for _, ix := range embeds {
|
||||
x = Implicit(DotField(pos, x, ix))
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue