Skip to content

Commit ff4091f

Browse files
committed
Clean up 'foreach' handling to match invocation handling
1 parent 653580d commit ff4091f

File tree

1 file changed

+32
-25
lines changed

1 file changed

+32
-25
lines changed

src/Roslyn.Diagnostics.Analyzers/Core/DoNotCopyValue.cs

+32-25
Original file line numberDiff line numberDiff line change
@@ -547,38 +547,45 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
547547
CheckLocalSymbolInUnsupportedContext(operation, local);
548548
}
549549

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;
551553
var instance2 = (operation.Collection as IConversionOperation)?.Operand;
552554

553-
switch (Acquire(operation.Collection))
555+
if (instance2 is null)
554556
{
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;
567559
}
568-
569-
switch (Acquire(instance2))
560+
else if (instance?.Conversion is not { IsIdentity: true, MethodSymbol: null })
570561
{
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);
578573

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;
580587
instance2 = null;
581-
break;
588+
}
582589
}
583590

584591
using var releaser = TryAddForVisit(_handledOperations, instance, out _);

0 commit comments

Comments
 (0)