Skip to content

Commit 653580d

Browse files
committedOct 27, 2020
Support foreach over parameters in DoNotCopyValue
1 parent a235f3f commit 653580d

File tree

2 files changed

+83
-6
lines changed

2 files changed

+83
-6
lines changed
 

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

+30-6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@
99
using Analyzer.Utilities;
1010
using Analyzer.Utilities.Extensions;
1111
using Microsoft.CodeAnalysis;
12+
using Microsoft.CodeAnalysis.CSharp;
13+
using Microsoft.CodeAnalysis.CSharp.Syntax;
1214
using Microsoft.CodeAnalysis.Diagnostics;
1315
using Microsoft.CodeAnalysis.FlowAnalysis;
1416
using Microsoft.CodeAnalysis.Operations;
@@ -367,7 +369,7 @@ bool IsSupportedConversion()
367369
switch (Acquire(operation.Operand))
368370
{
369371
case RefKind.None:
370-
case RefKind.Ref when operation.Conversion.IsIdentity:
372+
case RefKind.Ref or RefKind.RefReadOnly when operation.Conversion.IsIdentity:
371373
return true;
372374

373375
default:
@@ -547,14 +549,36 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
547549

548550
var instance = operation.Collection;
549551
var instance2 = (operation.Collection as IConversionOperation)?.Operand;
550-
if (Acquire(operation.Collection) != RefKind.Ref)
552+
553+
switch (Acquire(operation.Collection))
551554
{
552-
instance = null;
553-
instance2 = null;
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;
554567
}
555-
else if (Acquire(instance2) != RefKind.Ref)
568+
569+
switch (Acquire(instance2))
556570
{
557-
instance2 = null;
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;
578+
579+
default:
580+
instance2 = null;
581+
break;
558582
}
559583

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

‎src/Roslyn.Diagnostics.Analyzers/UnitTests/DoNotCopyValueTests.cs

+53
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
#nullable enable
44

55
using System.Threading.Tasks;
6+
using Microsoft.CodeAnalysis.Testing;
67
using Xunit;
78
using VerifyCS = Test.Utilities.CSharpSecurityCodeFixVerifier<
89
Roslyn.Diagnostics.Analyzers.DoNotCopyValue,
@@ -685,6 +686,58 @@ internal sealed class NonCopyableAttribute : System.Attribute { }
685686
}.RunAsync();
686687
}
687688

689+
[Theory]
690+
[CombinatorialData]
691+
public async Task AllowCustomForeachEnumeratorParameterReference(
692+
[CombinatorialValues("", "ref", "in")] string parameterModifiers,
693+
[CombinatorialValues("", "readonly")] string getEnumeratorModifiers)
694+
{
695+
var source = $@"
696+
using System.Runtime.InteropServices;
697+
698+
class C
699+
{{
700+
void Method({parameterModifiers} CannotCopy cannotCopy)
701+
{{
702+
foreach (var obj in {{|#0:cannotCopy|}})
703+
{{
704+
}}
705+
}}
706+
}}
707+
708+
[NonCopyable]
709+
struct CannotCopy
710+
{{
711+
public {getEnumeratorModifiers} Enumerator GetEnumerator() => throw null;
712+
713+
public struct Enumerator
714+
{{
715+
public object Current => throw null;
716+
public bool MoveNext() => throw null;
717+
}}
718+
}}
719+
720+
internal sealed class NonCopyableAttribute : System.Attribute {{ }}
721+
";
722+
723+
var expected = (parameterModifiers, getEnumeratorModifiers) switch
724+
{
725+
// /0/Test0.cs(8,29): warning RS0042: Unsupported use of non-copyable type 'CannotCopy' in 'ParameterReference' operation
726+
("in", "") => new[] { VerifyCS.Diagnostic(DoNotCopyValue.UnsupportedUseRule).WithLocation(0).WithArguments("CannotCopy", "ParameterReference") },
727+
728+
_ => DiagnosticResult.EmptyDiagnosticResults,
729+
};
730+
731+
var test = new VerifyCS.Test
732+
{
733+
TestCode = source,
734+
LanguageVersion = Microsoft.CodeAnalysis.CSharp.LanguageVersion.CSharp8,
735+
};
736+
737+
test.ExpectedDiagnostics.AddRange(expected);
738+
await test.RunAsync();
739+
}
740+
688741
[Fact]
689742
public async Task AllowCustomForeachEnumeratorDisposableObject()
690743
{

0 commit comments

Comments
 (0)