Skip to content

Commit 8982993

Browse files
authored
Merge pull request #4368 from sharwell/foreach-copy
Initial support for IForEachLoopOperation in DoNotCopyValue
2 parents 01ebd22 + 9f68083 commit 8982993

File tree

2 files changed

+141
-2
lines changed

2 files changed

+141
-2
lines changed

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

+22-2
Original file line numberDiff line numberDiff line change
@@ -364,9 +364,14 @@ bool IsSupportedConversion()
364364
return false;
365365
}
366366

367-
if (Acquire(operation.Operand) == RefKind.None)
367+
switch (Acquire(operation.Operand))
368368
{
369-
return true;
369+
case RefKind.None:
370+
case RefKind.Ref when operation.Conversion.IsIdentity:
371+
return true;
372+
373+
default:
374+
break;
370375
}
371376

372377
return false;
@@ -540,6 +545,21 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
540545
CheckLocalSymbolInUnsupportedContext(operation, local);
541546
}
542547

548+
var instance = operation.Collection;
549+
var instance2 = (operation.Collection as IConversionOperation)?.Operand;
550+
if (Acquire(operation.Collection) != RefKind.Ref)
551+
{
552+
instance = null;
553+
instance2 = null;
554+
}
555+
else if (Acquire(instance2) != RefKind.Ref)
556+
{
557+
instance2 = null;
558+
}
559+
560+
using var releaser = TryAddForVisit(_handledOperations, instance, out _);
561+
using var releaser2 = TryAddForVisit(_handledOperations, instance2, out _);
562+
543563
CheckTypeInUnsupportedContext(operation);
544564
base.VisitForEachLoop(operation);
545565
}

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

+119
Original file line numberDiff line numberDiff line change
@@ -636,6 +636,125 @@ struct CannotCopy
636636
public int Second { get; set; }
637637
}
638638
639+
internal sealed class NonCopyableAttribute : System.Attribute { }
640+
";
641+
642+
await new VerifyCS.Test
643+
{
644+
TestCode = source,
645+
LanguageVersion = Microsoft.CodeAnalysis.CSharp.LanguageVersion.CSharp8,
646+
}.RunAsync();
647+
}
648+
649+
[Fact]
650+
public async Task AllowCustomForeachEnumerator()
651+
{
652+
var source = @"
653+
using System.Runtime.InteropServices;
654+
655+
class C
656+
{
657+
void Method()
658+
{
659+
var cannotCopy = new CannotCopy();
660+
foreach (var obj in cannotCopy)
661+
{
662+
}
663+
}
664+
}
665+
666+
[NonCopyable]
667+
struct CannotCopy
668+
{
669+
public Enumerator GetEnumerator() => throw null;
670+
671+
public struct Enumerator
672+
{
673+
public object Current => throw null;
674+
public bool MoveNext() => throw null;
675+
}
676+
}
677+
678+
internal sealed class NonCopyableAttribute : System.Attribute { }
679+
";
680+
681+
await new VerifyCS.Test
682+
{
683+
TestCode = source,
684+
LanguageVersion = Microsoft.CodeAnalysis.CSharp.LanguageVersion.CSharp8,
685+
}.RunAsync();
686+
}
687+
688+
[Fact]
689+
public async Task AllowCustomForeachEnumeratorDisposableObject()
690+
{
691+
var source = @"
692+
using System;
693+
using System.Runtime.InteropServices;
694+
695+
class C
696+
{
697+
void Method()
698+
{
699+
using var cannotCopy = new CannotCopy();
700+
foreach (var obj in cannotCopy)
701+
{
702+
}
703+
}
704+
}
705+
706+
[NonCopyable]
707+
struct CannotCopy : IDisposable
708+
{
709+
public void Dispose() => throw null;
710+
public Enumerator GetEnumerator() => throw null;
711+
712+
public struct Enumerator
713+
{
714+
public object Current => throw null;
715+
public bool MoveNext() => throw null;
716+
}
717+
}
718+
719+
internal sealed class NonCopyableAttribute : System.Attribute { }
720+
";
721+
722+
await new VerifyCS.Test
723+
{
724+
TestCode = source,
725+
LanguageVersion = Microsoft.CodeAnalysis.CSharp.LanguageVersion.CSharp8,
726+
}.RunAsync();
727+
}
728+
729+
[Fact]
730+
public async Task AllowCustomForeachReadonlyEnumerator()
731+
{
732+
var source = @"
733+
using System.Runtime.InteropServices;
734+
735+
class C
736+
{
737+
void Method()
738+
{
739+
var cannotCopy = new CannotCopy();
740+
foreach (var obj in cannotCopy)
741+
{
742+
}
743+
}
744+
}
745+
746+
[NonCopyable]
747+
struct CannotCopy
748+
{
749+
public readonly Enumerator GetEnumerator() => throw null;
750+
751+
public struct Enumerator
752+
{
753+
public object Current => throw null;
754+
public bool MoveNext() => throw null;
755+
}
756+
}
757+
639758
internal sealed class NonCopyableAttribute : System.Attribute { }
640759
";
641760

0 commit comments

Comments
 (0)