@@ -547,38 +547,45 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
547
547
CheckLocalSymbolInUnsupportedContext ( operation , local ) ;
548
548
}
549
549
550
- var instance = operation . Collection ;
550
+ // 'foreach' operations have an identity conversion for the collection property, and then invoke the
551
+ // GetEnumerator method.
552
+ var instance = operation . Collection as IConversionOperation ;
551
553
var instance2 = ( operation . Collection as IConversionOperation ) ? . Operand ;
552
554
553
- switch ( Acquire ( operation . Collection ) )
555
+ if ( instance2 is null )
554
556
{
555
- case RefKind . Ref :
556
- // No special requirements
557
- break ;
558
-
559
- case RefKind . RefReadOnly when operation . Syntax is CommonForEachStatementSyntax syntax && operation . SemanticModel . GetForEachStatementInfo ( syntax ) . GetEnumeratorMethod is { IsReadOnly : true } :
560
- // Requirement of readonly GetEnumerator is met
561
- break ;
562
-
563
- default :
564
- instance = null ;
565
- instance2 = null ;
566
- break ;
557
+ // Didn't match the known pattern
558
+ instance = null ;
567
559
}
568
-
569
- switch ( Acquire ( instance2 ) )
560
+ else if ( instance ? . Conversion is not { IsIdentity : true , MethodSymbol : null } )
570
561
{
571
- case RefKind . Ref :
572
- // No special requirements
573
- break ;
574
-
575
- case RefKind . RefReadOnly when operation . Syntax is CommonForEachStatementSyntax syntax && operation . SemanticModel . GetForEachStatementInfo ( syntax ) . GetEnumeratorMethod is { IsReadOnly : true } :
576
- // Requirement of readonly GetEnumerator is met
577
- break ;
562
+ // Not a supported conversion
563
+ instance = null ;
564
+ instance2 = null ;
565
+ }
566
+ else
567
+ {
568
+ // Treat this as an invocation of the GetEnumerator method.
569
+ if ( operation . Syntax is CommonForEachStatementSyntax syntax
570
+ && operation . SemanticModel . GetForEachStatementInfo ( syntax ) . GetEnumeratorMethod is { } getEnumeratorMethod )
571
+ {
572
+ CheckMethodSymbolInUnsupportedContext ( operation , getEnumeratorMethod ) ;
578
573
579
- default :
574
+ if ( instance2 is not null
575
+ && _cache . IsNonCopyableType ( getEnumeratorMethod . ReceiverType )
576
+ && ! getEnumeratorMethod . IsReadOnly
577
+ && Acquire ( instance ) == RefKind . In )
578
+ {
579
+ // mark the instance as not checked by this method
580
+ instance2 = null ;
581
+ }
582
+ }
583
+ else
584
+ {
585
+ // Not supported
586
+ instance = null ;
580
587
instance2 = null ;
581
- break ;
588
+ }
582
589
}
583
590
584
591
using var releaser = TryAddForVisit ( _handledOperations , instance , out _ ) ;
0 commit comments