]> Cypherpunks repositories - gostls13.git/commitdiff
[dev.typeparams] cmd/compile: deal with inferred type arguments
authorDan Scales <danscales@google.com>
Tue, 2 Feb 2021 21:04:16 +0000 (13:04 -0800)
committerDan Scales <danscales@google.com>
Wed, 3 Feb 2021 02:15:51 +0000 (02:15 +0000)
Create an extra OFUNCINST node as needed, if there are inferred type
arguments for a generic function call.

Change-Id: Id990c5bcbce2893377072a7e41c7c6785d1eab60
Reviewed-on: https://go-review.googlesource.com/c/go/+/288952
Run-TryBot: Dan Scales <danscales@google.com>
TryBot-Result: Go Bot <gobot@golang.org>
Trust: Dan Scales <danscales@google.com>
Trust: Robert Griesemer <gri@golang.org>
Reviewed-by: Robert Griesemer <gri@golang.org>
src/cmd/compile/internal/noder/expr.go
src/cmd/compile/internal/noder/helpers.go

index 3c18bdcc2428f5e46fb12dbb5521044af6c5ecb3..568ec216e3050ac641a3e32cca48f88acf57348a 100644 (file)
@@ -93,11 +93,27 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
        case *syntax.AssertExpr:
                return Assert(pos, g.expr(expr.X), g.typeExpr(expr.Type))
        case *syntax.CallExpr:
-               def := g.info.Inferred[expr]
-               if len(def.Targs) > 0 {
-                       panic("Inferred type arguments not handled yet")
+               fun := g.expr(expr.Fun)
+               if inferred, ok := g.info.Inferred[expr]; ok && len(inferred.Targs) > 0 {
+                       targs := make([]ir.Node, len(inferred.Targs))
+                       for i, targ := range inferred.Targs {
+                               targs[i] = ir.TypeNode(g.typ(targ))
+                       }
+                       if fun.Op() == ir.OFUNCINST {
+                               // Replace explicit type args with the full list that
+                               // includes the additional inferred type args
+                               fun.(*ir.InstExpr).Targs = targs
+                       } else {
+                               // Create a function instantiation here, given
+                               // there are only inferred type args (e.g.
+                               // min(5,6), where min is a generic function)
+                               inst := ir.NewInstExpr(pos, ir.OFUNCINST, fun, targs)
+                               typed(fun.Type(), inst)
+                               fun = inst
+                       }
+
                }
-               return Call(pos, g.typ(typ), g.expr(expr.Fun), g.exprs(expr.ArgList), expr.HasDots)
+               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 {
@@ -111,7 +127,7 @@ func (g *irgen) expr0(typ types2.Type, expr syntax.Expr) ir.Node {
                        // This is generic function instantiation with a single type
                        targs = []ir.Node{index}
                }
-               // This is a generic function instantiation
+               // This is a generic function instantiation (e.g. min[int])
                x := g.expr(expr.X)
                if x.Op() != ir.ONAME || x.Type().Kind() != types.TFUNC {
                        panic("Incorrect argument for generic func instantiation")
index fcbb3a6ce5ee7e2e087219a94a04a4da9c0bc973..bb17a5331ad6265c50f67bde57c1a1bf55358674 100644 (file)
@@ -119,7 +119,7 @@ func Call(pos src.XPos, typ *types.Type, fun ir.Node, args []ir.Node, dots bool)
        n := ir.NewCallExpr(pos, ir.OCALL, fun, args)
        n.IsDDD = dots
 
-       if n.X.Op() != ir.OFUNCINST {
+       if fun.Op() != ir.OFUNCINST {
                // If no type params, still do normal typechecking, since we're
                // still missing some things done by tcCall below (mainly
                // typecheckargs and typecheckaste).