File: src\Workspaces\SharedUtilitiesAndExtensions\Compiler\Core\Extensions\OperationExtensions.cs
Web Access
Project: src\src\Workspaces\Core\Portable\Microsoft.CodeAnalysis.Workspaces.csproj (Microsoft.CodeAnalysis.Workspaces)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System;
using System.Collections.Immutable;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using Microsoft.CodeAnalysis.FlowAnalysis;
using Microsoft.CodeAnalysis.Operations;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis;
 
internal static partial class OperationExtensions
{
    public static bool IsTargetOfObjectMemberInitializer(this IOperation operation)
        => operation.Parent is IAssignmentOperation assignmentOperation &&
           assignmentOperation.Target == operation &&
           assignmentOperation.Parent?.Kind == OperationKind.ObjectOrCollectionInitializer;
 
    /// <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               |      |  ✔️   |             |             |                 |
        | obj is { } x             |      |  ✔️   |             |             |                 |
        | obj is [] x              |      |  ✔️   |             |             |                 |
        | ref var x =              |      |       |     ✔️      |     ✔️      |                 |
        | ref readonly var x =     |      |       |     ✔️      |             |                 |
 
        */
        // Workaround for https://github.com/dotnet/roslyn/issues/30753
        if (operation is ILocalReferenceOperation { IsDeclaration: true, IsImplicit: false })
        {
            // Declaration expression is a definition (write) for the declared local.
            return ValueUsageInfo.Write;
        }
        else if (operation is IDeclarationPatternOperation)
        {
            while (operation.Parent is IBinaryPatternOperation or INegatedPatternOperation or IRelationalPatternOperation)
                operation = operation.Parent;
 
            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;
            }
        }
        else if (operation is IRecursivePatternOperation or IListPatternOperation)
        {
            return ValueUsageInfo.Write;
        }
 
        if (operation.Parent is IAssignmentOperation assignmentOperation &&
            assignmentOperation.Target == operation)
        {
            return operation.Parent.IsAnyCompoundAssignment()
                ? ValueUsageInfo.ReadWrite
                : ValueUsageInfo.Write;
        }
        else if (operation.Parent is ISimpleAssignmentOperation { IsRef: true } simpleAssignmentOperation &&
            simpleAssignmentOperation.Value == operation)
        {
            return ValueUsageInfo.ReadableWritableReference;
        }
        else if (operation.Parent is IIncrementOrDecrementOperation)
        {
            return ValueUsageInfo.ReadWrite;
        }
        else if (operation.Parent is IForToLoopOperation forToLoopOperation && forToLoopOperation.LoopControlVariable.Equals(operation))
        {
            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)
        {
            switch (argumentOperation.Parameter?.RefKind)
            {
                case RefKind.RefReadOnly:
                    return ValueUsageInfo.ReadableReference;
 
                case RefKind.Out:
                    return ValueUsageInfo.WritableReference;
 
                case RefKind.Ref:
                    return ValueUsageInfo.ReadableWritableReference;
 
                default:
                    return 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 is IReDimOperation { Preserve: true }
                ? ValueUsageInfo.ReadWrite
                : ValueUsageInfo.Write;
        }
        else if (operation.Parent is IDeclarationExpressionOperation declarationExpression)
        {
            return declarationExpression.GetValueUsageInfo(containingSymbol);
        }
        else if (operation.IsInLeftOfDeconstructionAssignment(out _))
        {
            return ValueUsageInfo.Write;
        }
        else if (operation is { Type.IsValueType: true, Parent: IPropertyReferenceOperation })
        {
            // accessing an indexer/property off of a value type will read/write the value type depending on how the
            // indexer/property itself is used.
            return GetValueUsageInfo(operation.Parent, containingSymbol);
        }
        else if (operation.Parent is IVariableInitializerOperation variableInitializerOperation)
        {
            if (variableInitializerOperation.Parent is IVariableDeclaratorOperation variableDeclaratorOperation)
            {
                switch (variableDeclaratorOperation.Symbol.RefKind)
                {
                    case RefKind.Ref:
                        return ValueUsageInfo.ReadableWritableReference;
 
                    case RefKind.RefReadOnly:
                        return ValueUsageInfo.ReadableReference;
                }
            }
        }
 
        return ValueUsageInfo.Read;
    }
 
    public static RefKind GetRefKind(this IReturnOperation? operation, ISymbol containingSymbol)
    {
        var containingMethod = TryGetContainingAnonymousFunctionOrLocalFunction(operation) ?? (containingSymbol as IMethodSymbol);
        return containingMethod?.RefKind ?? RefKind.None;
    }
 
    public static IMethodSymbol? TryGetContainingAnonymousFunctionOrLocalFunction(this IOperation? operation)
    {
        operation = operation?.Parent;
        while (operation != null)
        {
            switch (operation.Kind)
            {
                case OperationKind.AnonymousFunction:
                    return ((IAnonymousFunctionOperation)operation).Symbol;
 
                case OperationKind.LocalFunction:
                    return ((ILocalFunctionOperation)operation).Symbol;
            }
 
            operation = operation.Parent;
        }
 
        return null;
    }
 
    public static bool IsInLeftOfDeconstructionAssignment(this IOperation operation, [NotNullWhen(true)] out IDeconstructionAssignmentOperation? deconstructionAssignment)
    {
        deconstructionAssignment = null;
 
        var previousOperation = operation;
        var current = operation.Parent;
 
        while (current != null)
        {
            switch (current.Kind)
            {
                case OperationKind.DeconstructionAssignment:
                    deconstructionAssignment = (IDeconstructionAssignmentOperation)current;
                    return deconstructionAssignment.Target == previousOperation;
 
                case OperationKind.Tuple:
                case OperationKind.Conversion:
                case OperationKind.Parenthesized:
                    previousOperation = current;
                    current = current.Parent;
                    continue;
 
                default:
                    return false;
            }
        }
 
        return false;
    }
 
    /// <summary>
    /// Returns true if the given operation is a regular compound assignment,
    /// i.e. <see cref="ICompoundAssignmentOperation"/> such as <code>a += b</code>,
    /// or a special null coalescing compound assignment, i.e. <see cref="ICoalesceAssignmentOperation"/>
    /// such as <code>a ??= b</code>.
    /// </summary>
    public static bool IsAnyCompoundAssignment(this IOperation operation)
    {
        switch (operation)
        {
            case ICompoundAssignmentOperation:
            case ICoalesceAssignmentOperation:
                return true;
 
            default:
                return false;
        }
    }
 
    public static bool IsInsideCatchRegion(this IOperation operation, ControlFlowGraph cfg)
    {
        foreach (var block in cfg.Blocks)
        {
            var isCatchRegionBlock = false;
            var currentRegion = block.EnclosingRegion;
            while (currentRegion != null)
            {
                switch (currentRegion.Kind)
                {
                    case ControlFlowRegionKind.Catch:
                        isCatchRegionBlock = true;
                        break;
                }
 
                currentRegion = currentRegion.EnclosingRegion;
            }
 
            if (isCatchRegionBlock)
            {
                foreach (var descendant in block.DescendantOperations())
                {
                    if (operation == descendant)
                    {
                        return true;
                    }
                }
            }
        }
 
        return false;
    }
 
    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)
        => operationBlock.HasAnyOperationDescendant(predicate, out _);
 
    public static bool HasAnyOperationDescendant(this IOperation operationBlock, Func<IOperation, bool> predicate, [NotNullWhen(true)] out IOperation? foundOperation)
    {
        RoslynDebug.AssertNotNull(operationBlock);
        RoslynDebug.AssertNotNull(predicate);
        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)
        => operationBlocks.HasAnyOperationDescendant(predicate: operation => operation.Kind == kind);
 
    public static bool IsNumericLiteral(this IOperation operation)
        => operation.Kind == OperationKind.Literal && operation.Type.IsNumericType();
 
    public static bool IsNullLiteral(this IOperation operand)
        => operand is ILiteralOperation { ConstantValue: { HasValue: true, Value: null } };
 
    /// <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>
    [return: NotNullIfNotNull(nameof(operation))]
    public static IOperation? WalkDownConversion(this IOperation? operation)
    {
        while (operation is IConversionOperation conversionOperation)
        {
            operation = conversionOperation.Operand;
        }
 
        return operation;
    }
 
    public static bool IsSingleThrowNotImplementedOperation([NotNullWhen(true)] this IOperation? firstBlock)
    {
        if (firstBlock is null)
            return false;
 
        var compilation = firstBlock.SemanticModel!.Compilation;
        var notImplementedExceptionType = compilation.NotImplementedExceptionType();
        if (notImplementedExceptionType == null)
            return false;
 
        if (firstBlock is not IBlockOperation block)
            return false;
 
        if (block.Operations.Length == 0)
            return false;
 
        var firstOp = block.Operations.Length == 1
            ? block.Operations[0]
            : TryGetSingleExplicitStatement(block.Operations);
        if (firstOp == null)
            return false;
 
        if (firstOp is IExpressionStatementOperation expressionStatement)
        {
            // unwrap: { throw new NYI(); }
            firstOp = expressionStatement.Operation;
        }
        else if (firstOp is IReturnOperation returnOperation)
        {
            // unwrap: 'int M(int p) => throw new NYI();'
            // For this case, the throw operation is wrapped within a conversion operation to 'int',
            // which in turn is wrapped within a return operation.
            firstOp = returnOperation.ReturnedValue.WalkDownConversion();
        }
 
        // => throw new NotImplementedOperation(...)
        return IsThrowNotImplementedOperation(notImplementedExceptionType, firstOp);
 
        static IOperation? TryGetSingleExplicitStatement(ImmutableArray<IOperation> operations)
        {
            IOperation? firstOp = null;
            foreach (var operation in operations)
            {
                if (operation.IsImplicit)
                    continue;
 
                if (firstOp != null)
                    return null;
 
                firstOp = operation;
            }
 
            return firstOp;
        }
 
        static bool IsThrowNotImplementedOperation(INamedTypeSymbol notImplementedExceptionType, IOperation? operation)
            => operation is IThrowOperation throwOperation &&
               throwOperation.Exception.UnwrapImplicitConversion() is IObjectCreationOperation objectCreation &&
               notImplementedExceptionType.Equals(objectCreation.Type);
    }
 
    [return: NotNullIfNotNull(nameof(value))]
    public static IOperation? UnwrapImplicitConversion(this IOperation? value)
        => value is IConversionOperation conversion && conversion.IsImplicit
            ? conversion.Operand
            : value;
}