9
9
using Analyzer . Utilities ;
10
10
using Analyzer . Utilities . Extensions ;
11
11
using Microsoft . CodeAnalysis ;
12
- using Microsoft . CodeAnalysis . CSharp ;
13
- using Microsoft . CodeAnalysis . CSharp . Syntax ;
14
12
using Microsoft . CodeAnalysis . Diagnostics ;
15
13
using Microsoft . CodeAnalysis . FlowAnalysis ;
16
14
using Microsoft . CodeAnalysis . Operations ;
17
15
18
16
namespace Roslyn . Diagnostics . Analyzers
19
17
{
20
- [ DiagnosticAnalyzer ( LanguageNames . CSharp , LanguageNames . VisualBasic ) ]
21
- public sealed class DoNotCopyValue : DiagnosticAnalyzer
18
+ public abstract class AbstractDoNotCopyValue : DiagnosticAnalyzer
22
19
{
23
20
private static readonly LocalizableString s_localizableTitle = new LocalizableResourceString ( nameof ( RoslynDiagnosticsAnalyzersResources . DoNotCopyValueTitle ) , RoslynDiagnosticsAnalyzersResources . ResourceManager , typeof ( RoslynDiagnosticsAnalyzersResources ) ) ;
24
21
private static readonly LocalizableString s_localizableMessage = new LocalizableResourceString ( nameof ( RoslynDiagnosticsAnalyzersResources . DoNotCopyValueMessage ) , RoslynDiagnosticsAnalyzersResources . ResourceManager , typeof ( RoslynDiagnosticsAnalyzersResources ) ) ;
@@ -93,6 +90,8 @@ public sealed class DoNotCopyValue : DiagnosticAnalyzer
93
90
94
91
public override ImmutableArray < DiagnosticDescriptor > SupportedDiagnostics => ImmutableArray . Create ( Rule , UnsupportedUseRule , NoBoxingRule , NoUnboxingRule ) ;
95
92
93
+ protected abstract NonCopyableWalker CreateWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache ) ;
94
+
96
95
public override void Initialize ( AnalysisContext context )
97
96
{
98
97
context . EnableConcurrentExecution ( ) ;
@@ -105,9 +104,9 @@ public override void Initialize(AnalysisContext context)
105
104
} ) ;
106
105
}
107
106
108
- private static void AnalyzeOperationBlock ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
107
+ private void AnalyzeOperationBlock ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
109
108
{
110
- var walker = new NonCopyableWalker ( context , cache ) ;
109
+ var walker = CreateWalker ( context , cache ) ;
111
110
foreach ( var operation in context . OperationBlocks )
112
111
{
113
112
walker . Visit ( operation ) ;
@@ -149,18 +148,21 @@ public void Dispose()
149
148
}
150
149
}
151
150
152
- private sealed class NonCopyableWalker : OperationWalker
151
+ protected abstract class NonCopyableWalker : OperationWalker
153
152
{
154
153
private readonly OperationBlockAnalysisContext _context ;
155
- private readonly NonCopyableTypesCache _cache ;
156
154
private readonly HashSet < IOperation > _handledOperations = new HashSet < IOperation > ( ) ;
157
155
158
- public NonCopyableWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
156
+ protected NonCopyableWalker ( OperationBlockAnalysisContext context , NonCopyableTypesCache cache )
159
157
{
160
158
_context = context ;
161
- _cache = cache ;
159
+ Cache = cache ;
162
160
}
163
161
162
+ protected NonCopyableTypesCache Cache { get ; }
163
+
164
+ protected abstract bool CheckForEachGetEnumerator ( IForEachLoopOperation operation , [ DisallowNull ] ref IConversionOperation ? conversion , [ DisallowNull ] ref IOperation ? instance ) ;
165
+
164
166
public override void VisitAddressOf ( IAddressOfOperation operation )
165
167
{
166
168
CheckTypeInUnsupportedContext ( operation ) ;
@@ -222,11 +224,11 @@ public override void VisitAwait(IAwaitOperation operation)
222
224
{
223
225
// Treat await of ValueTask<T> the same way handling of a return
224
226
if ( operation . Type is { } type
225
- && _cache . IsNonCopyableType ( type )
227
+ && Cache . IsNonCopyableType ( type )
226
228
&& operation . Operation . Type is INamedTypeSymbol { OriginalDefinition : var taskType } )
227
229
{
228
- if ( ! SymbolEqualityComparer . Default . Equals ( taskType , _cache . ValueTaskT )
229
- && ! SymbolEqualityComparer . Default . Equals ( taskType , _cache . ConfiguredValueTaskAwaitableT ) )
230
+ if ( ! SymbolEqualityComparer . Default . Equals ( taskType , Cache . ValueTaskT )
231
+ && ! SymbolEqualityComparer . Default . Equals ( taskType , Cache . ConfiguredValueTaskAwaitableT ) )
230
232
{
231
233
CheckTypeInUnsupportedContext ( operation ) ;
232
234
}
@@ -566,21 +568,7 @@ public override void VisitForEachLoop(IForEachLoopOperation operation)
566
568
else
567
569
{
568
570
// 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 ) ;
573
-
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
571
+ if ( ! CheckForEachGetEnumerator ( operation , ref instance , ref instance2 ) )
584
572
{
585
573
// Not supported
586
574
instance = null ;
@@ -669,7 +657,7 @@ public override void VisitInvocation(IInvocationOperation operation)
669
657
670
658
var instance = operation . Instance ;
671
659
if ( instance is object
672
- && _cache . IsNonCopyableType ( operation . TargetMethod . ReceiverType )
660
+ && Cache . IsNonCopyableType ( operation . TargetMethod . ReceiverType )
673
661
&& ! operation . TargetMethod . IsReadOnly
674
662
&& Acquire ( instance ) == RefKind . In )
675
663
{
@@ -837,7 +825,7 @@ public override void VisitPropertyReference(IPropertyReferenceOperation operatio
837
825
838
826
var instance = operation . Instance ;
839
827
if ( instance is object
840
- && _cache . IsNonCopyableType ( operation . Property . ContainingType )
828
+ && Cache . IsNonCopyableType ( operation . Property . ContainingType )
841
829
&& Acquire ( instance ) == RefKind . In )
842
830
{
843
831
if ( operation . IsSetMethodInvocation ( ) )
@@ -1144,7 +1132,7 @@ private static bool CanAssign(RefKind sourceRefKind, RefKind targetRefKind)
1144
1132
} ;
1145
1133
}
1146
1134
1147
- private RefKind Acquire ( IOperation ? operation )
1135
+ protected RefKind Acquire ( IOperation ? operation )
1148
1136
{
1149
1137
if ( operation is null )
1150
1138
return RefKind . RefReadOnly ;
@@ -1338,7 +1326,7 @@ private void CheckTypeSymbolInUnsupportedContext(IOperation operation, ITypeSymb
1338
1326
if ( type . OriginalDefinition . SpecialType == SpecialType . System_Nullable_T )
1339
1327
{
1340
1328
var nullableUnderlyingType = ( ( INamedTypeSymbol ) type ) . TypeArguments . FirstOrDefault ( ) ;
1341
- if ( _cache . IsNonCopyableType ( nullableUnderlyingType ) )
1329
+ if ( Cache . IsNonCopyableType ( nullableUnderlyingType ) )
1342
1330
{
1343
1331
_context . ReportDiagnostic ( operation . Syntax . CreateDiagnostic ( AvoidNullableWrapperRule , type , operation . Kind ) ) ;
1344
1332
}
@@ -1380,7 +1368,7 @@ private void CheckLocalSymbolInUnsupportedContext(IOperation operation, ILocalSy
1380
1368
CheckTypeSymbolInUnsupportedContext ( operation , local . Type ) ;
1381
1369
}
1382
1370
1383
- private void CheckMethodSymbolInUnsupportedContext ( IOperation operation , IMethodSymbol ? symbol )
1371
+ protected void CheckMethodSymbolInUnsupportedContext ( IOperation operation , IMethodSymbol ? symbol )
1384
1372
{
1385
1373
if ( symbol is null )
1386
1374
return ;
@@ -1424,7 +1412,7 @@ private void CheckTypeInUnsupportedContext(IOperation operation)
1424
1412
return ;
1425
1413
}
1426
1414
1427
- if ( ! _cache . IsNonCopyableType ( symbol ) )
1415
+ if ( ! Cache . IsNonCopyableType ( symbol ) )
1428
1416
{
1429
1417
// Copies of this type are allowed
1430
1418
return ;
@@ -1434,7 +1422,7 @@ private void CheckTypeInUnsupportedContext(IOperation operation)
1434
1422
}
1435
1423
}
1436
1424
1437
- private sealed class NonCopyableTypesCache
1425
+ protected sealed class NonCopyableTypesCache
1438
1426
{
1439
1427
private readonly ConcurrentDictionary < INamedTypeSymbol , bool > _typesToNonCopyable
1440
1428
= new ConcurrentDictionary < INamedTypeSymbol , bool > ( ) ;
0 commit comments