@@ -321,8 +321,8 @@ func (check *Checker) callExpr(x *operand, call *syntax.CallExpr) exprKind {
321
321
}
322
322
323
323
// evaluate arguments
324
- args := check .genericExprList (call .ArgList )
325
- sig = check .arguments (call , sig , targs , xlist , args )
324
+ args , atargs , atxlist := check .genericExprList (call .ArgList )
325
+ sig = check .arguments (call , sig , targs , xlist , args , atargs , atxlist )
326
326
327
327
if wasGeneric && sig .TypeParams ().Len () == 0 {
328
328
// update the recorded type of call.Fun to its instantiated type
@@ -358,65 +358,123 @@ func (check *Checker) callExpr(x *operand, call *syntax.CallExpr) exprKind {
358
358
359
359
// exprList evaluates a list of expressions and returns the corresponding operands.
360
360
// A single-element expression list may evaluate to multiple operands.
361
- func (check * Checker ) exprList (elist []syntax.Expr ) []* operand {
362
- switch len (elist ) {
363
- case 0 :
364
- return nil
365
-
366
- case 1 :
367
- xlist , _ := check .multiExpr (elist [0 ], false )
368
- return xlist
369
-
370
- default :
361
+ func (check * Checker ) exprList (elist []syntax.Expr ) (xlist []* operand ) {
362
+ if n := len (elist ); n == 1 {
363
+ xlist , _ = check .multiExpr (elist [0 ], false )
364
+ } else if n > 1 {
371
365
// multiple (possibly invalid) values
372
- xlist : = make ([]* operand , len ( elist ) )
366
+ xlist = make ([]* operand , n )
373
367
for i , e := range elist {
374
368
var x operand
375
369
check .expr (nil , & x , e )
376
370
xlist [i ] = & x
377
371
}
378
- return xlist
379
372
}
373
+ return
380
374
}
381
375
382
- // genericExprList is like exprList but result operands may be generic (not fully instantiated).
383
- func (check * Checker ) genericExprList (elist []syntax.Expr ) []* operand {
384
- switch len (elist ) {
385
- case 0 :
386
- return nil
376
+ // genericExprList is like exprList but result operands may be uninstantiated or partially
377
+ // instantiated generic functions.
378
+ // For each non-generic or uninstantiated generic operand, the corresponding targsList and
379
+ // xlistList elements do not exist (targsList and xlistList are nil) or the elements are nil.
380
+ // For each partially instantiated generic function operand, the corresponding targsList and
381
+ // xlistList elements are the operand's partial type arguments and type expression lists.
382
+ func (check * Checker ) genericExprList (elist []syntax.Expr ) (resList []* operand , targsList [][]Type , xlistList [][]syntax.Expr ) {
383
+ if debug {
384
+ defer func () {
385
+ // targsList and xlistList must have matching lengths
386
+ assert (len (targsList ) == len (xlistList ))
387
+ // type arguments must only exist for partially instantiated functions
388
+ for i , x := range resList {
389
+ if i < len (targsList ) {
390
+ if n := len (targsList [i ]); n > 0 {
391
+ // x must be a partially instantiated function
392
+ assert (n < x .typ .(* Signature ).TypeParams ().Len ())
393
+ }
394
+ }
395
+ }
396
+ }()
397
+ }
387
398
388
- case 1 :
399
+ if n := len (elist ); n == 1 {
400
+ // single value (possibly a partially instantiated function), or a multi-valued expression
389
401
e := elist [0 ]
390
402
var x operand
391
- check .rawExpr (nil , & x , e , nil , true )
392
- check .exclude (& x , 1 << novalue | 1 << builtin | 1 << typexpr )
393
-
394
- if t , ok := x .typ .(* Tuple ); ok && x .mode != invalid {
395
- // multiple values - cannot be generic
396
- xlist := make ([]* operand , t .Len ())
397
- for i , v := range t .vars {
398
- xlist [i ] = & operand {mode : value , expr : e , typ : v .typ }
403
+ if inst , _ := e .(* syntax.IndexExpr ); inst != nil && check .indexExpr (& x , inst ) {
404
+ // x is a generic function.
405
+ targs , xlist := check .funcInst (nil , x .Pos (), & x , inst , false )
406
+ if targs != nil {
407
+ // x was not instantiated: collect the (partial) type arguments.
408
+ targsList = [][]Type {targs }
409
+ xlistList = [][]syntax.Expr {xlist }
410
+ // Update x.expr so that we can record the partially instantiated function.
411
+ x .expr = inst
412
+ } else {
413
+ // x was instantiated: we must record it here because we didn't
414
+ // use the usual expression evaluators.
415
+ check .record (& x )
416
+ }
417
+ resList = []* operand {& x }
418
+ } else {
419
+ // x is not a function instantiation (it may still be a generic function).
420
+ check .rawExpr (nil , & x , e , nil , true )
421
+ check .exclude (& x , 1 << novalue | 1 << builtin | 1 << typexpr )
422
+ if t , ok := x .typ .(* Tuple ); ok && x .mode != invalid {
423
+ // x is a function call returning multiple values; it cannot be generic.
424
+ resList = make ([]* operand , t .Len ())
425
+ for i , v := range t .vars {
426
+ resList [i ] = & operand {mode : value , expr : e , typ : v .typ }
427
+ }
428
+ } else {
429
+ // x is exactly one value (possibly invalid or uninstantiated generic function).
430
+ resList = []* operand {& x }
399
431
}
400
- return xlist
401
432
}
402
-
403
- // exactly one (possible invalid or generic) value
404
- return []* operand {& x }
405
-
406
- default :
407
- // multiple (possibly invalid) values
408
- xlist := make ([]* operand , len (elist ))
433
+ } else if n > 1 {
434
+ // multiple values
435
+ resList = make ([]* operand , n )
436
+ targsList = make ([][]Type , n )
437
+ xlistList = make ([][]syntax.Expr , n )
409
438
for i , e := range elist {
410
439
var x operand
411
- check .genericExpr (& x , e )
412
- xlist [i ] = & x
440
+ if inst , _ := e .(* syntax.IndexExpr ); inst != nil && check .indexExpr (& x , inst ) {
441
+ // x is a generic function.
442
+ targs , xlist := check .funcInst (nil , x .Pos (), & x , inst , false )
443
+ if targs != nil {
444
+ // x was not instantiated: collect the (partial) type arguments.
445
+ targsList [i ] = targs
446
+ xlistList [i ] = xlist
447
+ // Update x.expr so that we can record the partially instantiated function.
448
+ x .expr = inst
449
+ } else {
450
+ // x was instantiated: we must record it here because we didn't
451
+ // use the usual expression evaluators.
452
+ check .record (& x )
453
+ }
454
+ } else {
455
+ // x is exactly one value (possibly invalid or uninstantiated generic function).
456
+ check .genericExpr (& x , e )
457
+ }
458
+ resList [i ] = & x
413
459
}
414
- return xlist
415
460
}
461
+
462
+ return
416
463
}
417
464
418
- // xlist is the list of type argument expressions supplied in the source code.
419
- func (check * Checker ) arguments (call * syntax.CallExpr , sig * Signature , targs []Type , xlist []syntax.Expr , args []* operand ) (rsig * Signature ) {
465
+ // arguments type-checks arguments passed to a function call with the given signature.
466
+ // The function and its arguments may be generic, and possibly partially instantiated.
467
+ // targs and xlist are the function's type arguments (and corresponding expressions).
468
+ // args are the function arguments. If an argument args[i] is a partially instantiated
469
+ // generic function, atargs[i] and atxlist[i] are the corresponding type arguments
470
+ // (and corresponding expressions).
471
+ // If the callee is variadic, arguments adjusts its signature to match the provided
472
+ // arguments. The type parameters and arguments of the callee and all its arguments
473
+ // are used together to infer any missing type arguments, and the callee and argument
474
+ // functions are instantiated as necessary.
475
+ // The result signature is the (possibly adjusted and instantiated) function signature.
476
+ // If an error occured, the result signature is the incoming sig.
477
+ func (check * Checker ) arguments (call * syntax.CallExpr , sig * Signature , targs []Type , xlist []syntax.Expr , args []* operand , atargs [][]Type , atxlist [][]syntax.Expr ) (rsig * Signature ) {
420
478
rsig = sig
421
479
422
480
// Function call argument/parameter count requirements
@@ -516,7 +574,12 @@ func (check *Checker) arguments(call *syntax.CallExpr, sig *Signature, targs []T
516
574
var tmp Type
517
575
tparams , tmp = check .renameTParams (call .Pos (), sig .TypeParams ().list (), sigParams )
518
576
sigParams = tmp .(* Tuple )
577
+ // make sure targs and tparams have the same length
578
+ for len (targs ) < len (tparams ) {
579
+ targs = append (targs , nil )
580
+ }
519
581
}
582
+ assert (len (tparams ) == len (targs ))
520
583
521
584
// collect type parameters from generic function arguments
522
585
var genericArgs []int // indices of generic function arguments
@@ -533,10 +596,20 @@ func (check *Checker) arguments(call *syntax.CallExpr, sig *Signature, targs []T
533
596
asig .tparams = & TypeParamList {atparams } // renameTParams doesn't touch associated type parameters
534
597
arg .typ = asig // new type identity for the function argument
535
598
tparams = append (tparams , atparams ... )
599
+ // add partial list of type arguments, if any
600
+ if i < len (atargs ) {
601
+ targs = append (targs , atargs [i ]... )
602
+ }
603
+ // make sure targs and tparams have the same length
604
+ for len (targs ) < len (tparams ) {
605
+ targs = append (targs , nil )
606
+ }
536
607
genericArgs = append (genericArgs , i )
537
608
}
538
609
}
539
610
}
611
+ assert (len (tparams ) == len (targs ))
612
+
540
613
// at the moment we only support implicit instantiations of argument functions
541
614
_ = len (genericArgs ) > 0 && check .verifyVersionf (check .pkg , args [genericArgs [0 ]], go1_21 , "implicitly instantiated function as argument" )
542
615
@@ -574,11 +647,12 @@ func (check *Checker) arguments(call *syntax.CallExpr, sig *Signature, targs []T
574
647
// compute argument signatures: instantiate if needed
575
648
j := n
576
649
for _ , i := range genericArgs {
577
- asig := args [i ].typ .(* Signature )
650
+ arg := args [i ]
651
+ asig := arg .typ .(* Signature )
578
652
k := j + asig .TypeParams ().Len ()
579
653
// targs[j:k] are the inferred type arguments for asig
580
- asig = check .instantiateSignature (call .Pos (), args [ i ] .expr , asig , targs [j :k ], nil ) // TODO(gri) provide xlist if possible (partial instantiations)
581
- args [ i ]. typ = asig
654
+ arg . typ = check .instantiateSignature (call .Pos (), arg .expr , asig , targs [j :k ], nil ) // TODO(gri) provide xlist if possible (partial instantiations)
655
+ check . record ( arg ) // record here because we didn't use the usual expr evaluators
582
656
j = k
583
657
}
584
658
}
0 commit comments