-
Notifications
You must be signed in to change notification settings - Fork 467
/
IOperationExtensions.cs
1096 lines (947 loc) · 47.2 KB
/
IOperationExtensions.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// Copyright (c) Microsoft. All Rights Reserved. Licensed under the MIT license. See License.txt in the project root for license information.
#if HAS_IOPERATION
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using System.Threading;
using Analyzer.Utilities.PooledObjects;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.FlowAnalysis;
using Microsoft.CodeAnalysis.Operations;
namespace Analyzer.Utilities.Extensions
{
internal static partial class IOperationExtensions
{
/// <summary>
/// Gets the receiver type for an invocation expression (i.e. type of 'A' in invocation 'A.B()')
/// If the invocation actually involves a conversion from A to some other type, say 'C', on which B is invoked,
/// then this method returns type A if <paramref name="beforeConversion"/> is true, and C if false.
/// </summary>
public static INamedTypeSymbol? GetReceiverType(this IInvocationOperation invocation, Compilation compilation, bool beforeConversion, CancellationToken cancellationToken)
{
if (invocation.Instance != null)
{
return beforeConversion ?
GetReceiverType(invocation.Instance.Syntax, compilation, cancellationToken) :
invocation.Instance.Type as INamedTypeSymbol;
}
else if (invocation.TargetMethod.IsExtensionMethod && !invocation.TargetMethod.Parameters.IsEmpty)
{
var firstArg = invocation.Arguments.FirstOrDefault();
if (firstArg != null)
{
return beforeConversion ?
GetReceiverType(firstArg.Value.Syntax, compilation, cancellationToken) :
firstArg.Type as INamedTypeSymbol;
}
else if (invocation.TargetMethod.Parameters[0].IsParams)
{
return invocation.TargetMethod.Parameters[0].Type as INamedTypeSymbol;
}
}
return null;
}
private static INamedTypeSymbol? GetReceiverType(SyntaxNode receiverSyntax, Compilation compilation, CancellationToken cancellationToken)
{
var model = compilation.GetSemanticModel(receiverSyntax.SyntaxTree);
var typeInfo = model.GetTypeInfo(receiverSyntax, cancellationToken);
return typeInfo.Type as INamedTypeSymbol;
}
public static bool HasNullConstantValue(this IOperation operation)
{
return operation.ConstantValue.HasValue && operation.ConstantValue.Value == null;
}
public static bool TryGetBoolConstantValue(this IOperation operation, out bool constantValue)
{
if (operation.ConstantValue.HasValue && operation.ConstantValue.Value is bool value)
{
constantValue = value;
return true;
}
constantValue = false;
return false;
}
public static bool HasConstantValue(this IOperation operation, long comparand)
{
return operation.HasConstantValue(unchecked((ulong)comparand));
}
public static bool HasConstantValue(this IOperation operation, ulong comparand)
{
var constantValue = operation.ConstantValue;
if (!constantValue.HasValue)
{
return false;
}
if (operation.Type == null || operation.Type.IsErrorType())
{
return false;
}
if (operation.Type.IsPrimitiveType())
{
return HasConstantValue(constantValue, operation.Type, comparand);
}
if (operation.Type.TypeKind == TypeKind.Enum)
{
var enumUnderlyingType = ((INamedTypeSymbol)operation.Type).EnumUnderlyingType;
return enumUnderlyingType != null &&
enumUnderlyingType.IsPrimitiveType() &&
HasConstantValue(constantValue, enumUnderlyingType, comparand);
}
return false;
}
private static bool HasConstantValue(Optional<object?> constantValue, ITypeSymbol constantValueType, ulong comparand)
{
if (constantValueType.SpecialType is SpecialType.System_Double or SpecialType.System_Single)
{
return (double?)constantValue.Value == comparand;
}
return DiagnosticHelpers.TryConvertToUInt64(constantValue.Value, constantValueType.SpecialType, out ulong convertedValue) && convertedValue == comparand;
}
public static ITypeSymbol? GetElementType(this IArrayCreationOperation? arrayCreation)
{
return (arrayCreation?.Type as IArrayTypeSymbol)?.ElementType;
}
/// <summary>
/// Filters out operations that are implicit and have no explicit descendant with a constant value or a non-null type.
/// </summary>
public static ImmutableArray<IOperation> WithoutFullyImplicitOperations(this ImmutableArray<IOperation> operations)
{
ImmutableArray<IOperation>.Builder? builder = null;
for (int i = 0; i < operations.Length; i++)
{
var operation = operations[i];
// Check if all descendants are either implicit or are explicit with no constant value or type, indicating it is not user written code.
if (operation.DescendantsAndSelf().All(o => o.IsImplicit || (!o.ConstantValue.HasValue && o.Type == null)))
{
if (builder == null)
{
builder = ImmutableArray.CreateBuilder<IOperation>();
builder.AddRange(operations, i);
}
}
else if (builder != null)
{
builder.Add(operation);
}
}
return builder != null ? builder.ToImmutable() : operations;
}
/// <summary>
/// Gets explicit descendants or self of the given <paramref name="operation"/> that have no explicit ancestor in
/// the operation tree rooted at <paramref name="operation"/>.
/// </summary>
/// <param name="operation">Operation</param>
public static ImmutableArray<IOperation> GetTopmostExplicitDescendants(this IOperation operation)
{
if (!operation.IsImplicit)
{
return ImmutableArray.Create(operation);
}
var builder = ImmutableArray.CreateBuilder<IOperation>();
var operationsToProcess = new Queue<IOperation>();
operationsToProcess.Enqueue(operation);
while (operationsToProcess.Count > 0)
{
operation = operationsToProcess.Dequeue();
if (!operation.IsImplicit)
{
builder.Add(operation);
}
else
{
foreach (var child in operation.Children)
{
operationsToProcess.Enqueue(child);
}
}
}
return builder.ToImmutable();
}
/// <summary>
/// True if this operation has no IOperation API support, i.e. <see cref="OperationKind.None"/> and
/// is the root operation, i.e. <see cref="Operation.Parent"/> is null.
/// For example, this returns true for attribute operations.
/// </summary>
public static bool IsOperationNoneRoot(this IOperation operation)
{
return operation.Kind == OperationKind.None && operation.Parent == null;
}
/// <summary>
/// Returns the topmost <see cref="IBlockOperation"/> containing the given <paramref name="operation"/>.
/// </summary>
public static IBlockOperation? GetTopmostParentBlock(this IOperation? operation)
{
IOperation? currentOperation = operation;
IBlockOperation? topmostBlockOperation = null;
while (currentOperation != null)
{
if (currentOperation is IBlockOperation blockOperation)
{
topmostBlockOperation = blockOperation;
}
currentOperation = currentOperation.Parent;
}
return topmostBlockOperation;
}
/// <summary>
/// Gets the first ancestor of this operation with:
/// 1. Specified OperationKind
/// 2. If <paramref name="predicate"/> is non-null, it succeeds for the ancestor.
/// Returns null if there is no such ancestor.
/// </summary>
public static TOperation? GetAncestor<TOperation>(this IOperation root, OperationKind ancestorKind, Func<TOperation, bool>? predicate = null)
where TOperation : class, IOperation
{
if (root == null)
{
throw new ArgumentNullException(nameof(root));
}
var ancestor = root;
do
{
ancestor = ancestor.Parent;
} while (ancestor != null && ancestor.Kind != ancestorKind);
if (ancestor != null)
{
if (predicate != null && !predicate((TOperation)ancestor))
{
return GetAncestor(ancestor, ancestorKind, predicate);
}
return (TOperation)ancestor;
}
else
{
return default;
}
}
/// <summary>
/// Gets the first ancestor of this operation with:
/// 1. Any OperationKind from the specified <paramref name="ancestorKinds"/>.
/// 2. If <paramref name="predicate"/> is non-null, it succeeds for the ancestor.
/// Returns null if there is no such ancestor.
/// </summary>
public static IOperation? GetAncestor(this IOperation root, ImmutableArray<OperationKind> ancestorKinds, Func<IOperation, bool>? predicate = null)
{
if (root == null)
{
throw new ArgumentNullException(nameof(root));
}
var ancestor = root;
do
{
ancestor = ancestor.Parent;
} while (ancestor != null && !ancestorKinds.Contains(ancestor.Kind));
if (ancestor != null)
{
if (predicate != null && !predicate(ancestor))
{
return GetAncestor(ancestor, ancestorKinds, predicate);
}
return ancestor;
}
else
{
return default;
}
}
public static IConditionalAccessOperation? GetConditionalAccess(this IConditionalAccessInstanceOperation operation)
{
return operation.GetAncestor(OperationKind.ConditionalAccess, (IConditionalAccessOperation c) => c.Operation.Syntax == operation.Syntax);
}
/// <summary>
/// Gets the operation for the object being created that is being referenced by <paramref name="operation"/>.
/// If the operation is referencing an implicit or an explicit this/base/Me/MyBase/MyClass instance, then we return "null".
/// </summary>
/// <param name="operation"></param>
/// <param name="isInsideAnonymousObjectInitializer">Flag to indicate if the operation is a descendant of an <see cref="IAnonymousObjectCreationOperation"/>.</param>
/// <remarks>
/// PERF: Note that the parameter <paramref name="isInsideAnonymousObjectInitializer"/> is to improve performance by avoiding walking the entire IOperation parent for non-initializer cases.
/// </remarks>
public static IOperation? GetInstance(this IInstanceReferenceOperation operation, bool isInsideAnonymousObjectInitializer)
{
Debug.Assert(isInsideAnonymousObjectInitializer ==
(operation.GetAncestor<IAnonymousObjectCreationOperation>(OperationKind.AnonymousObjectCreation) != null));
if (isInsideAnonymousObjectInitializer)
{
for (IOperation? current = operation; current != null && current.Kind != OperationKind.Block; current = current.Parent)
{
switch (current.Kind)
{
// VB object initializer allows accessing the members of the object being created with "." operator.
// The syntax of such an IInstanceReferenceOperation points to the object being created.
// Check for such an IAnonymousObjectCreationOperation with matching syntax.
// For example, instance reference for members ".Field1" and ".Field2" in "New C() With { .Field1 = 0, .Field2 = .Field1 }".
case OperationKind.AnonymousObjectCreation:
if (current.Syntax == operation.Syntax)
{
return current;
}
break;
}
}
}
// For all other cases, IInstanceReferenceOperation refers to the implicit or explicit this/base/Me/MyBase/MyClass reference.
// We return null for such cases.
return null;
}
public static bool HasAnyOperationDescendant(this ImmutableArray<IOperation> operationBlocks, Func<IOperation, bool> predicate)
{
foreach (var operationBlock in operationBlocks)
{
if (operationBlock.HasAnyOperationDescendant(predicate))
{
return true;
}
}
return false;
}
public static bool HasAnyOperationDescendant(this IOperation operationBlock, Func<IOperation, bool> predicate)
{
return operationBlock.HasAnyOperationDescendant(predicate, out _);
}
public static bool HasAnyOperationDescendant(this IOperation operationBlock, Func<IOperation, bool> predicate, [NotNullWhen(returnValue: true)] out IOperation? foundOperation)
{
foreach (var descendant in operationBlock.DescendantsAndSelf())
{
if (predicate(descendant))
{
foundOperation = descendant;
return true;
}
}
foundOperation = null;
return false;
}
public static bool HasAnyOperationDescendant(this ImmutableArray<IOperation> operationBlocks, OperationKind kind)
{
return operationBlocks.HasAnyOperationDescendant(predicate: operation => operation.Kind == kind);
}
/// <summary>
/// Indicates if the given <paramref name="binaryOperation"/> is a predicate operation used in a condition.
/// </summary>
/// <param name="binaryOperation"></param>
/// <returns></returns>
public static bool IsComparisonOperator(this IBinaryOperation binaryOperation)
=> binaryOperation.OperatorKind switch
{
BinaryOperatorKind.Equals
or BinaryOperatorKind.NotEquals
or BinaryOperatorKind.ObjectValueEquals
or BinaryOperatorKind.ObjectValueNotEquals
or BinaryOperatorKind.LessThan
or BinaryOperatorKind.LessThanOrEqual
or BinaryOperatorKind.GreaterThan
or BinaryOperatorKind.GreaterThanOrEqual => true,
_ => false,
};
public static IOperation GetRoot(this IOperation operation)
{
while (operation.Parent != null)
{
operation = operation.Parent;
}
return operation;
}
/// <summary>
/// PERF: Cache from operation roots to their corresponding <see cref="ControlFlowGraph"/> to enable interprocedural flow analysis
/// across analyzers and analyzer callbacks to re-use the control flow graph.
/// </summary>
/// <remarks>Also see <see cref="IMethodSymbolExtensions.s_methodToTopmostOperationBlockCache"/></remarks>
private static readonly BoundedCache<Compilation, ConcurrentDictionary<IOperation, ControlFlowGraph?>> s_operationToCfgCache
= new();
public static bool TryGetEnclosingControlFlowGraph(this IOperation operation, [NotNullWhen(returnValue: true)] out ControlFlowGraph? cfg)
{
operation = operation.GetRoot();
RoslynDebug.Assert(operation.SemanticModel is not null);
var operationToCfgMap = s_operationToCfgCache.GetOrCreateValue(operation.SemanticModel.Compilation);
cfg = operationToCfgMap.GetOrAdd(operation, CreateControlFlowGraph);
return cfg != null;
}
public static ControlFlowGraph? GetEnclosingControlFlowGraph(this IBlockOperation blockOperation)
{
var success = blockOperation.TryGetEnclosingControlFlowGraph(out var cfg);
Debug.Assert(success);
Debug.Assert(cfg != null);
return cfg;
}
private static ControlFlowGraph? CreateControlFlowGraph(IOperation operation)
{
switch (operation)
{
case IBlockOperation blockOperation:
return ControlFlowGraph.Create(blockOperation);
case IMethodBodyOperation methodBodyOperation:
return ControlFlowGraph.Create(methodBodyOperation);
case IConstructorBodyOperation constructorBodyOperation:
return ControlFlowGraph.Create(constructorBodyOperation);
case IFieldInitializerOperation fieldInitializerOperation:
return ControlFlowGraph.Create(fieldInitializerOperation);
case IPropertyInitializerOperation propertyInitializerOperation:
return ControlFlowGraph.Create(propertyInitializerOperation);
case IParameterInitializerOperation:
// We do not support flow analysis for parameter initializers
return null;
default:
// Attribute blocks have OperationKind.None, but ControlFlowGraph.Create does not
// have an overload for such operation roots.
// Gracefully return null for this case and fire an assert for any other OperationKind.
Debug.Assert(operation.Kind == OperationKind.None, $"Unexpected root operation kind: {operation.Kind}");
return null;
}
}
/// <summary>
/// Gets the symbols captured from the enclosing function(s) by the given lambda or local function.
/// </summary>
/// <param name="operation">Operation representing the lambda or local function.</param>
/// <param name="lambdaOrLocalFunction">Method symbol for the lambda or local function.</param>
public static PooledHashSet<ISymbol> GetCaptures(this IOperation operation, IMethodSymbol lambdaOrLocalFunction)
{
Debug.Assert(operation is IAnonymousFunctionOperation anonymousFunction && anonymousFunction.Symbol.OriginalDefinition.ReturnTypeAndParametersAreSame(lambdaOrLocalFunction.OriginalDefinition) ||
operation is ILocalFunctionOperation localFunction && localFunction.Symbol.OriginalDefinition.Equals(lambdaOrLocalFunction.OriginalDefinition));
lambdaOrLocalFunction = lambdaOrLocalFunction.OriginalDefinition;
var builder = PooledHashSet<ISymbol>.GetInstance();
using var nestedLambdasAndLocalFunctions = PooledHashSet<IMethodSymbol>.GetInstance();
nestedLambdasAndLocalFunctions.Add(lambdaOrLocalFunction);
foreach (var child in operation.Descendants())
{
switch (child.Kind)
{
case OperationKind.LocalReference:
ProcessLocalOrParameter(((ILocalReferenceOperation)child).Local);
break;
case OperationKind.ParameterReference:
ProcessLocalOrParameter(((IParameterReferenceOperation)child).Parameter);
break;
case OperationKind.InstanceReference:
builder.Add(lambdaOrLocalFunction.ContainingType);
break;
case OperationKind.AnonymousFunction:
nestedLambdasAndLocalFunctions.Add(((IAnonymousFunctionOperation)child).Symbol);
break;
case OperationKind.LocalFunction:
nestedLambdasAndLocalFunctions.Add(((ILocalFunctionOperation)child).Symbol);
break;
}
}
return builder;
// Local functions.
void ProcessLocalOrParameter(ISymbol symbol)
{
if (symbol.ContainingSymbol?.Kind == SymbolKind.Method &&
!nestedLambdasAndLocalFunctions.Contains(symbol.ContainingSymbol.OriginalDefinition))
{
builder.Add(symbol);
}
}
}
private static readonly ImmutableArray<OperationKind> s_LambdaAndLocalFunctionKinds =
ImmutableArray.Create(OperationKind.AnonymousFunction, OperationKind.LocalFunction);
public static bool IsWithinLambdaOrLocalFunction(this IOperation operation, [NotNullWhen(true)] out IOperation? containingLambdaOrLocalFunctionOperation)
{
containingLambdaOrLocalFunctionOperation = operation.GetAncestor(s_LambdaAndLocalFunctionKinds);
return containingLambdaOrLocalFunctionOperation != null;
}
public static bool IsWithinExpressionTree(this IOperation operation, [NotNullWhen(true)] INamedTypeSymbol? linqExpressionTreeType)
=> linqExpressionTreeType != null
&& operation.GetAncestor(s_LambdaAndLocalFunctionKinds)?.Parent?.Type?.OriginalDefinition is { } lambdaType
&& linqExpressionTreeType.Equals(lambdaType);
public static ITypeSymbol? GetPatternType(this IPatternOperation pattern)
{
return pattern switch
{
#if CODEANALYSIS_V3_OR_BETTER
IDeclarationPatternOperation declarationPattern => declarationPattern.MatchedType,
IRecursivePatternOperation recursivePattern => recursivePattern.MatchedType,
IDiscardPatternOperation discardPattern => discardPattern.InputType,
#else
IDeclarationPatternOperation declarationPattern => declarationPattern.DeclaredSymbol switch
{
ILocalSymbol local => local.Type,
IDiscardSymbol discard => discard.Type,
_ => null,
},
#endif
IConstantPatternOperation constantPattern => constantPattern.Value.Type,
_ => null,
};
}
/// <summary>
/// If the given <paramref name="tupleOperation"/> is a nested tuple,
/// gets the parenting tuple operation and the tuple element of that parenting tuple
/// which contains the given tupleOperation as a descendant operation.
/// </summary>
public static bool TryGetParentTupleOperation(this ITupleOperation tupleOperation,
[NotNullWhen(returnValue: true)] out ITupleOperation? parentTupleOperation,
[NotNullWhen(returnValue: true)] out IOperation? elementOfParentTupleContainingTuple)
{
parentTupleOperation = null;
elementOfParentTupleContainingTuple = null;
IOperation previousOperation = tupleOperation;
var currentOperation = tupleOperation.Parent;
while (currentOperation != null)
{
switch (currentOperation.Kind)
{
case OperationKind.Parenthesized:
case OperationKind.Conversion:
case OperationKind.DeclarationExpression:
previousOperation = currentOperation;
currentOperation = currentOperation.Parent;
continue;
case OperationKind.Tuple:
parentTupleOperation = (ITupleOperation)currentOperation;
elementOfParentTupleContainingTuple = previousOperation;
return true;
default:
return false;
}
}
return false;
}
public static bool IsExtensionMethodAndHasNoInstance(this IInvocationOperation invocationOperation)
{
// This method exists to abstract away the language specific differences between IInvocationOperation implementations
// See https://github.com/dotnet/roslyn/issues/23625 for more details
return invocationOperation.TargetMethod.IsExtensionMethod && (invocationOperation.Language != LanguageNames.VisualBasic || invocationOperation.Instance == null);
}
public static IOperation? GetInstance(this IInvocationOperation invocationOperation)
=> invocationOperation.IsExtensionMethodAndHasNoInstance() ? invocationOperation.Arguments[0].Value : invocationOperation.Instance;
public static SyntaxNode? GetInstanceSyntax(this IInvocationOperation invocationOperation)
=> invocationOperation.GetInstance()?.Syntax;
public static ITypeSymbol? GetInstanceType(this IOperation operation)
{
IOperation? instance = operation switch
{
IInvocationOperation invocation => invocation.GetInstance(),
IPropertyReferenceOperation propertyReference => propertyReference.Instance,
_ => throw new NotImplementedException()
};
return instance?.WalkDownConversion().Type;
}
public static ISymbol? GetReferencedMemberOrLocalOrParameter(this IOperation operation)
{
return operation switch
{
IMemberReferenceOperation memberReference => memberReference.Member,
IParameterReferenceOperation parameterReference => parameterReference.Parameter,
ILocalReferenceOperation localReference => localReference.Local,
IParenthesizedOperation parenthesized => parenthesized.Operand.GetReferencedMemberOrLocalOrParameter(),
IConversionOperation conversion => conversion.Operand.GetReferencedMemberOrLocalOrParameter(),
_ => null,
};
}
/// <summary>
/// Walks down consecutive parenthesized operations until an operand is reached that isn't a parenthesized operation.
/// </summary>
/// <param name="operation">The starting operation.</param>
/// <returns>The inner non parenthesized operation or the starting operation if it wasn't a parenthesized operation.</returns>
public static IOperation WalkDownParentheses(this IOperation operation)
{
while (operation is IParenthesizedOperation parenthesizedOperation)
{
operation = parenthesizedOperation.Operand;
}
return operation;
}
[return: NotNullIfNotNull("operation")]
public static IOperation? WalkUpParentheses(this IOperation? operation)
{
if (operation is null)
return null;
while (operation.Parent is IParenthesizedOperation parenthesizedOperation)
{
operation = parenthesizedOperation;
}
return operation;
}
/// <summary>
/// Walks down consecutive conversion operations until an operand is reached that isn't a conversion operation.
/// </summary>
/// <param name="operation">The starting operation.</param>
/// <returns>The inner non conversion operation or the starting operation if it wasn't a conversion operation.</returns>
public static IOperation WalkDownConversion(this IOperation operation)
{
while (operation is IConversionOperation conversionOperation)
{
operation = conversionOperation.Operand;
}
return operation;
}
/// <summary>
/// Walks down consecutive conversion operations that satisfy <paramref name="predicate"/> until an operand is reached that
/// either isn't a conversion or doesn't satisfy <paramref name="predicate"/>.
/// </summary>
/// <param name="operation">The starting operation.</param>
/// <param name="predicate">A predicate to filter conversion operations.</param>
/// <returns>The first operation that either isn't a conversion or doesn't satisfy <paramref name="predicate"/>.</returns>
public static IOperation WalkDownConversion(this IOperation operation, Func<IConversionOperation, bool> predicate)
{
while (operation is IConversionOperation conversionOperation && predicate(conversionOperation))
{
operation = conversionOperation.Operand;
}
return operation;
}
[return: NotNullIfNotNull("operation")]
public static IOperation? WalkUpConversion(this IOperation? operation)
{
if (operation is null)
return null;
while (operation.Parent is IConversionOperation conversionOperation)
{
operation = conversionOperation;
}
return operation;
}
public static ITypeSymbol? GetThrownExceptionType(this IThrowOperation operation)
{
var thrownObject = operation.Exception;
// Starting C# 8.0, C# compiler wraps the thrown operation within an implicit conversion to System.Exception type.
if (thrownObject is IConversionOperation conversion &&
conversion.IsImplicit)
{
thrownObject = conversion.Operand;
}
return thrownObject?.Type;
}
/// <summary>
/// Determines if the one of the invocation's arguments' values is an argument of the specified type, and if so, find
/// the first one.
/// </summary>
/// <param name="invocationOperation">Invocation operation whose arguments to look through.</param>
/// <param name="firstFoundArgument">First found IArgumentOperation.Value of the specified type, order by the method's
/// signature's parameters (as opposed to how arguments are specified when invoked).</param>
/// <returns>True if one is found, false otherwise.</returns>
/// <remarks>
/// IInvocationOperation.Arguments are ordered by how they are specified, which may differ from the order in the method
/// signature if the caller specifies arguments by name. This will find the first typeof operation ordered by the
/// method signature's parameters.
/// </remarks>
public static bool HasArgument<TOperation>(
this IInvocationOperation invocationOperation,
[NotNullWhen(returnValue: true)] out TOperation? firstFoundArgument)
where TOperation : class, IOperation
{
firstFoundArgument = null;
int minOrdinal = int.MaxValue;
foreach (IArgumentOperation argumentOperation in invocationOperation.Arguments)
{
if (argumentOperation.Parameter?.Ordinal < minOrdinal && argumentOperation.Value is TOperation to)
{
minOrdinal = argumentOperation.Parameter.Ordinal;
firstFoundArgument = to;
}
}
return firstFoundArgument != null;
}
public static bool HasAnyExplicitDescendant(this IOperation operation, Func<IOperation, bool>? descendIntoOperation = null)
{
using var stack = ArrayBuilder<IEnumerator<IOperation>>.GetInstance();
stack.Add(operation.Children.GetEnumerator());
while (stack.Any())
{
var enumerator = stack.Last();
stack.RemoveLast();
if (enumerator.MoveNext())
{
var current = enumerator.Current;
stack.Add(enumerator);
if (current != null &&
(descendIntoOperation == null || descendIntoOperation(current)))
{
if (!current.IsImplicit &&
// This prevents non explicit operations like expression to be considered as ok
(current.ConstantValue.HasValue || current.Type != null))
{
return true;
}
stack.Add(current.Children.GetEnumerator());
}
}
}
return false;
}
public static bool IsSetMethodInvocation(this IPropertyReferenceOperation operation)
{
if (operation.Property.SetMethod is null)
{
// This is either invalid code, or an assignment through a ref-returning getter
return false;
}
IOperation potentialLeftSide = operation;
while (potentialLeftSide.Parent is IParenthesizedOperation or ITupleOperation)
{
potentialLeftSide = potentialLeftSide.Parent;
}
return potentialLeftSide.Parent switch
{
IAssignmentOperation { Target: var target } when target == potentialLeftSide => true,
_ => false,
};
}
public static IArgumentOperation GetArgumentForParameterAtIndex(
this ImmutableArray<IArgumentOperation> arguments,
int parameterIndex)
{
Debug.Assert(parameterIndex >= 0);
Debug.Assert(parameterIndex < arguments.Length);
foreach (var argument in arguments)
{
if (argument.Parameter?.Ordinal == parameterIndex)
{
return argument;
}
}
throw new InvalidOperationException();
}
/// <summary>
/// Useful when named arguments used for a method call and you need them in the original parameter order.
/// </summary>
/// <param name="arguments">Arguments of the method</param>
/// <returns>Returns the arguments in parameter order</returns>
public static ImmutableArray<IArgumentOperation> GetArgumentsInParameterOrder(
this ImmutableArray<IArgumentOperation> arguments)
{
using var parameterOrderedArguments = ArrayBuilder<IArgumentOperation>.GetInstance(arguments.Length, null!);
foreach (var argument in arguments)
{
RoslynDebug.Assert(argument.Parameter is not null);
Debug.Assert(parameterOrderedArguments[argument.Parameter.Ordinal] == null);
parameterOrderedArguments[argument.Parameter.Ordinal] = argument;
}
return parameterOrderedArguments.ToImmutableArray();
}
// Copied from roslyn https://github.com/dotnet/roslyn/blob/main/src/Workspaces/SharedUtilitiesAndExtensions/Compiler/Core/Extensions/OperationExtensions.cs#L25
#if CODEANALYSIS_V3_OR_BETTER
/// <summary>
/// Returns the <see cref="ValueUsageInfo"/> for the given operation.
/// This extension can be removed once https://github.com/dotnet/roslyn/issues/25057 is implemented.
/// </summary>
public static ValueUsageInfo GetValueUsageInfo(this IOperation operation, ISymbol containingSymbol)
{
/*
| code | Read | Write | ReadableRef | WritableRef | NonReadWriteRef |
| x.Prop = 1 | | ✔️ | | | |
| x.Prop += 1 | ✔️ | ✔️ | | | |
| x.Prop++ | ✔️ | ✔️ | | | |
| Foo(x.Prop) | ✔️ | | | | |
| Foo(x.Prop), | | | ✔️ | | |
where void Foo(in T v)
| Foo(out x.Prop) | | | | ✔️ | |
| Foo(ref x.Prop) | | | ✔️ | ✔️ | |
| nameof(x) | | | | | ✔️ | ️
| sizeof(x) | | | | | ✔️ | ️
| typeof(x) | | | | | ✔️ | ️
| out var x | | ✔️ | | | | ️
| case X x: | | ✔️ | | | | ️
| obj is X x | | ✔️ | | | |
| ref var x = | | | ✔️ | ✔️ | |
| ref readonly var x = | | | ✔️ | | |
*/
if (operation is ILocalReferenceOperation localReference &&
localReference.IsDeclaration &&
!localReference.IsImplicit) // Workaround for https://github.com/dotnet/roslyn/issues/30753
{
// Declaration expression is a definition (write) for the declared local.
return ValueUsageInfo.Write;
}
else if (operation is IDeclarationPatternOperation)
{
switch (operation.Parent)
{
case IPatternCaseClauseOperation:
// A declaration pattern within a pattern case clause is a
// write for the declared local.
// For example, 'x' is defined and assigned the value from 'obj' below:
// switch (obj)
// {
// case X x:
//
return ValueUsageInfo.Write;
case IRecursivePatternOperation:
// A declaration pattern within a recursive pattern is a
// write for the declared local.
// For example, 'x' is defined and assigned the value from 'obj' below:
// (obj) switch
// {
// (X x) => ...
// };
//
return ValueUsageInfo.Write;
case ISwitchExpressionArmOperation:
// A declaration pattern within a switch expression arm is a
// write for the declared local.
// For example, 'x' is defined and assigned the value from 'obj' below:
// obj switch
// {
// X x => ...
//
return ValueUsageInfo.Write;
case IIsPatternOperation:
// A declaration pattern within an is pattern is a
// write for the declared local.
// For example, 'x' is defined and assigned the value from 'obj' below:
// if (obj is X x)
//
return ValueUsageInfo.Write;
case IPropertySubpatternOperation:
// A declaration pattern within a property sub-pattern is a
// write for the declared local.
// For example, 'x' is defined and assigned the value from 'obj.Property' below:
// if (obj is { Property : int x })
//
return ValueUsageInfo.Write;
default:
Debug.Fail("Unhandled declaration pattern context");
// Conservatively assume read/write.
return ValueUsageInfo.ReadWrite;
}
}
if (operation.Parent is IAssignmentOperation assignmentOperation &&
assignmentOperation.Target == operation)
{
return operation.Parent.IsAnyCompoundAssignment()
? ValueUsageInfo.ReadWrite
: ValueUsageInfo.Write;
}
else if (operation.Parent is IIncrementOrDecrementOperation)
{
return ValueUsageInfo.ReadWrite;
}
else if (operation.Parent is IParenthesizedOperation parenthesizedOperation)
{
// Note: IParenthesizedOperation is specific to VB, where the parens cause a copy, so this cannot be classified as a write.
Debug.Assert(parenthesizedOperation.Language == LanguageNames.VisualBasic);
return parenthesizedOperation.GetValueUsageInfo(containingSymbol) &
~(ValueUsageInfo.Write | ValueUsageInfo.Reference);
}
else if (operation.Parent is INameOfOperation or
ITypeOfOperation or
ISizeOfOperation)
{
return ValueUsageInfo.Name;
}
else if (operation.Parent is IArgumentOperation argumentOperation)
{
return argumentOperation.Parameter?.RefKind switch
{
RefKind.RefReadOnly => ValueUsageInfo.ReadableReference,
RefKind.Out => ValueUsageInfo.WritableReference,
RefKind.Ref => ValueUsageInfo.ReadableWritableReference,
_ => ValueUsageInfo.Read,
};
}
else if (operation.Parent is IReturnOperation returnOperation)
{
return returnOperation.GetRefKind(containingSymbol) switch
{
RefKind.RefReadOnly => ValueUsageInfo.ReadableReference,
RefKind.Ref => ValueUsageInfo.ReadableWritableReference,
_ => ValueUsageInfo.Read,
};
}
else if (operation.Parent is IConditionalOperation conditionalOperation)
{
if (operation == conditionalOperation.WhenTrue
|| operation == conditionalOperation.WhenFalse)
{
return GetValueUsageInfo(conditionalOperation, containingSymbol);
}
else
{
return ValueUsageInfo.Read;
}
}
else if (operation.Parent is IReDimClauseOperation reDimClauseOperation &&
reDimClauseOperation.Operand == operation)
{
return (reDimClauseOperation.Parent as IReDimOperation)?.Preserve == true
? ValueUsageInfo.ReadWrite
: ValueUsageInfo.Write;
}
else if (operation.Parent is IDeclarationExpressionOperation declarationExpression)