15
15
16
16
namespace Roslyn . Diagnostics . Analyzers
17
17
{
18
- [ DiagnosticAnalyzer ( LanguageNames . CSharp , LanguageNames . VisualBasic ) ]
19
- public sealed class DoNotCopyValue : DiagnosticAnalyzer
18
+ public abstract class AbstractDoNotCopyValue : DiagnosticAnalyzer
20
19
{
21
20
private static readonly LocalizableString s_localizableTitle = new LocalizableResourceString ( nameof ( RoslynDiagnosticsAnalyzersResources . DoNotCopyValueTitle ) , RoslynDiagnosticsAnalyzersResources . ResourceManager , typeof ( RoslynDiagnosticsAnalyzersResources ) ) ;
22
21
private static readonly LocalizableString s_localizableMessage = new LocalizableResourceString ( nameof ( RoslynDiagnosticsAnalyzersResources . DoNotCopyValueMessage ) , RoslynDiagnosticsAnalyzersResources . ResourceManager , typeof ( RoslynDiagnosticsAnalyzersResources ) ) ;
@@ -91,6 +90,8 @@ public sealed class DoNotCopyValue : DiagnosticAnalyzer
91
90
92
91
public override ImmutableArray < DiagnosticDescriptor > SupportedDiagnostics => ImmutableArray . Create ( Rule , UnsupportedUseRule , NoBoxingRule , NoUnboxingRule ) ;
93
92
93
+ protected abstract NonCopyableWalker CreateWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache ) ;
94
+
94
95
public override void Initialize ( AnalysisContext context )
95
96
{
96
97
context . EnableConcurrentExecution ( ) ;
@@ -103,9 +104,9 @@ public override void Initialize(AnalysisContext context)
103
104
} ) ;
104
105
}
105
106
106
- private static void AnalyzeOperationBlock ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
107
+ private void AnalyzeOperationBlock ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
107
108
{
108
- var walker = new NonCopyableWalker ( context , cache ) ;
109
+ var walker = CreateWalker ( context , cache ) ;
109
110
foreach ( var operation in context . OperationBlocks )
110
111
{
111
112
walker . Visit ( operation ) ;
@@ -147,18 +148,21 @@ public void Dispose()
147
148
}
148
149
}
149
150
150
- private sealed class NonCopyableWalker : OperationWalker
151
+ protected abstract class NonCopyableWalker : OperationWalker
151
152
{
152
153
private readonly OperationBlockAnalysisContext _context ;
153
- private readonly NonCopyableTypesCache _cache ;
154
154
private readonly HashSet < IOperation > _handledOperations = new HashSet < IOperation > ( ) ;
155
155
156
- public NonCopyableWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
156
+ protected NonCopyableWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
157
157
{
158
158
_context = context ;
159
- _cache = cache ;
159
+ Cache = cache ;
160
160
}
161
161
162
+ protected NonCopyableTypesCache Cache { get ; }
163
+
164
+ protected abstract bool CheckForEachGetEnumerator ( IForEachLoopOperation operation , [ DisallowNull ] ref IConversionOperation ? conversion , [ DisallowNull ] ref IOperation ? instance ) ;
165
+
162
166
public override void VisitAddressOf ( IAddressOfOperation operation )
163
167
{
164
168
CheckTypeInUnsupportedContext ( operation ) ;
@@ -220,11 +224,11 @@ public override void VisitAwait(IAwaitOperation operation)
220
224
{
221
225
// Treat await of ValueTask<T> the same way handling of a return
222
226
if ( operation . Type is { } type
223
- && _cache . IsNonCopyableType ( type )
227
+ && Cache . IsNonCopyableType ( type )
224
228
&& operation . Operation . Type is INamedTypeSymbol { OriginalDefinition : var taskType } )
225
229
{
226
- if ( ! SymbolEqualityComparer . Default . Equals ( taskType , _cache . ValueTaskT )
227
- && ! SymbolEqualityComparer . Default . Equals ( taskType , _cache . ConfiguredValueTaskAwaitableT ) )
230
+ if ( ! SymbolEqualityComparer . Default . Equals ( taskType , Cache . ValueTaskT )
231
+ && ! SymbolEqualityComparer . Default . Equals ( taskType , Cache . ConfiguredValueTaskAwaitableT ) )
228
232
{
229
233
CheckTypeInUnsupportedContext ( operation ) ;
230
234
}
@@ -367,7 +371,7 @@ bool IsSupportedConversion()
367
371
switch ( Acquire ( operation . Operand ) )
368
372
{
369
373
case RefKind . None :
370
- case RefKind . Ref when operation . Conversion . IsIdentity :
374
+ case RefKind . Ref or RefKind . RefReadOnly when operation . Conversion . IsIdentity :
371
375
return true ;
372
376
373
377
default :
@@ -545,17 +549,32 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
545
549
CheckLocalSymbolInUnsupportedContext ( operation , local ) ;
546
550
}
547
551
548
- var instance = operation . Collection ;
552
+ // 'foreach' operations have an identity conversion for the collection property, and then invoke the
553
+ // GetEnumerator method.
554
+ var instance = operation . Collection as IConversionOperation ;
549
555
var instance2 = ( operation . Collection as IConversionOperation ) ? . Operand ;
550
- if ( Acquire ( operation . Collection ) != RefKind . Ref )
556
+
557
+ if ( instance2 is null )
551
558
{
559
+ // Didn't match the known pattern
552
560
instance = null ;
553
- instance2 = null ;
554
561
}
555
- else if ( Acquire ( instance2 ) != RefKind . Ref )
562
+ else if ( instance ? . Conversion is not { IsIdentity : true , MethodSymbol : null } )
556
563
{
564
+ // Not a supported conversion
565
+ instance = null ;
557
566
instance2 = null ;
558
567
}
568
+ else
569
+ {
570
+ // Treat this as an invocation of the GetEnumerator method.
571
+ if ( ! CheckForEachGetEnumerator ( operation , ref instance , ref instance2 ) )
572
+ {
573
+ // Not supported
574
+ instance = null ;
575
+ instance2 = null ;
576
+ }
577
+ }
559
578
560
579
using var releaser = TryAddForVisit ( _handledOperations , instance , out _ ) ;
561
580
using var releaser2 = TryAddForVisit ( _handledOperations , instance2 , out _ ) ;
@@ -638,7 +657,7 @@ public override void VisitInvocation(IInvocationOperation operation)
638
657
639
658
var instance = operation . Instance ;
640
659
if ( instance is object
641
- && _cache . IsNonCopyableType ( operation . TargetMethod . ReceiverType )
660
+ && Cache . IsNonCopyableType ( operation . TargetMethod . ReceiverType )
642
661
&& ! operation . TargetMethod . IsReadOnly
643
662
&& Acquire ( instance ) == RefKind . In )
644
663
{
@@ -806,7 +825,7 @@ public override void VisitPropertyReference(IPropertyReferenceOperation operatio
806
825
807
826
var instance = operation . Instance ;
808
827
if ( instance is object
809
- && _cache . IsNonCopyableType ( operation . Property . ContainingType )
828
+ && Cache . IsNonCopyableType ( operation . Property . ContainingType )
810
829
&& Acquire ( instance ) == RefKind . In )
811
830
{
812
831
if ( operation . IsSetMethodInvocation ( ) )
@@ -1113,7 +1132,7 @@ private static bool CanAssign(RefKind sourceRefKind, RefKind targetRefKind)
1113
1132
} ;
1114
1133
}
1115
1134
1116
- private RefKind Acquire ( IOperation ? operation )
1135
+ protected RefKind Acquire ( IOperation ? operation )
1117
1136
{
1118
1137
if ( operation is null )
1119
1138
return RefKind . RefReadOnly ;
@@ -1307,7 +1326,7 @@ private void CheckTypeSymbolInUnsupportedContext(IOperation operation, ITypeSymb
1307
1326
if ( type . OriginalDefinition . SpecialType == SpecialType . System_Nullable_T )
1308
1327
{
1309
1328
var nullableUnderlyingType = ( ( INamedTypeSymbol ) type ) . TypeArguments . FirstOrDefault ( ) ;
1310
- if ( _cache . IsNonCopyableType ( nullableUnderlyingType ) )
1329
+ if ( Cache . IsNonCopyableType ( nullableUnderlyingType ) )
1311
1330
{
1312
1331
_context . ReportDiagnostic ( operation . Syntax . CreateDiagnostic ( AvoidNullableWrapperRule , type , operation . Kind ) ) ;
1313
1332
}
@@ -1349,7 +1368,7 @@ private void CheckLocalSymbolInUnsupportedContext(IOperation operation, ILocalSy
1349
1368
CheckTypeSymbolInUnsupportedContext ( operation , local . Type ) ;
1350
1369
}
1351
1370
1352
- private void CheckMethodSymbolInUnsupportedContext ( IOperation operation , IMethodSymbol ? symbol )
1371
+ protected void CheckMethodSymbolInUnsupportedContext ( IOperation operation , IMethodSymbol ? symbol )
1353
1372
{
1354
1373
if ( symbol is null )
1355
1374
return ;
@@ -1393,7 +1412,7 @@ private void CheckTypeInUnsupportedContext(IOperation operation)
1393
1412
return ;
1394
1413
}
1395
1414
1396
- if ( ! _cache . IsNonCopyableType ( symbol ) )
1415
+ if ( ! Cache . IsNonCopyableType ( symbol ) )
1397
1416
{
1398
1417
// Copies of this type are allowed
1399
1418
return ;
@@ -1403,7 +1422,7 @@ private void CheckTypeInUnsupportedContext(IOperation operation)
1403
1422
}
1404
1423
}
1405
1424
1406
- private sealed class NonCopyableTypesCache
1425
+ protected sealed class NonCopyableTypesCache
1407
1426
{
1408
1427
private readonly ConcurrentDictionary < INamedTypeSymbol , bool > _typesToNonCopyable
1409
1428
= new ConcurrentDictionary < INamedTypeSymbol , bool > ( ) ;
0 commit comments