File: src\Analyzers\CSharp\Analyzers\UsePatternCombinators\AnalyzedPattern.cs
Web Access
Project: src\src\Features\CSharp\Portable\Microsoft.CodeAnalysis.CSharp.Features.csproj (Microsoft.CodeAnalysis.CSharp.Features)
// 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.Linq;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Operations;
using Roslyn.Utilities;
 
namespace Microsoft.CodeAnalysis.CSharp.UsePatternCombinators;
 
using static SyntaxFactory;
 
/// <summary>
/// Base class to represent a pattern constructed from various checks
/// </summary>
internal abstract class AnalyzedPattern
{
    public readonly IOperation Target;
 
    private AnalyzedPattern(IOperation target)
        => Target = target;
 
    /// <summary>
    /// Represents a type-pattern, constructed from an is-expression
    /// </summary>
    internal sealed class Type : AnalyzedPattern
    {
        private static readonly SyntaxAnnotation s_annotation = new();
 
        public static Type? TryCreate(BinaryExpressionSyntax binaryExpression, IIsTypeOperation operation)
        {
            Contract.ThrowIfNull(operation.SemanticModel);
            if (binaryExpression.Right is not TypeSyntax typeSyntax)
            {
                return null;
            }
 
            // We are coming from a type pattern, which likes to bind to types, but converting to
            // patters which like to bind to expressions. For example, given:
            //
            // if (T is C.X || T is Y) { }
            //
            // we would want to convert to:
            //
            // if (T is C.X or Y)
            //
            // In the first case the compiler will bind to types named C or Y that are in scope
            // but in the second it will also bind to a fields, methods etc. which for 'Y' changes
            // semantics, and for 'C.X' could be a compile error.
            //
            // So lets create a pattern syntax and make sure the result is the same
            var dummyStatement = ExpressionStatement(AssignmentExpression(
                SyntaxKind.SimpleAssignmentExpression,
                IdentifierName("_"),
                IsPatternExpression(
                    binaryExpression.Left,
                    ConstantPattern(ParenthesizedExpression(binaryExpression.Right.WithAdditionalAnnotations(s_annotation)))
                )
            ));
 
            if (operation.SemanticModel.TryGetSpeculativeSemanticModel(typeSyntax.SpanStart, dummyStatement, out var speculativeModel))
            {
                var originalInfo = operation.SemanticModel.GetTypeInfo(binaryExpression.Right);
                var newInfo = speculativeModel.GetTypeInfo(dummyStatement.GetAnnotatedNodes(s_annotation).Single());
                if (!originalInfo.Equals(newInfo))
                {
                    return null;
                }
            }
 
            return new Type(typeSyntax, operation.ValueOperand);
        }
 
        public readonly TypeSyntax TypeSyntax;
 
        private Type(TypeSyntax type, IOperation target) : base(target)
            => TypeSyntax = type;
    }
 
    /// <summary>
    /// Represents a source-pattern, constructed from C# patterns
    /// </summary>
    internal sealed class Source(PatternSyntax patternSyntax, IOperation target) : AnalyzedPattern(target)
    {
        public readonly PatternSyntax PatternSyntax = patternSyntax;
    }
 
    /// <summary>
    /// Represents a constant-pattern, constructed from an equality check
    /// </summary>
    internal sealed class Constant(ExpressionSyntax expression, IOperation target) : AnalyzedPattern(target)
    {
        public readonly ExpressionSyntax ExpressionSyntax = expression;
    }
 
    /// <summary>
    /// Represents a relational-pattern, constructed from relational operators
    /// </summary>
    internal sealed class Relational(BinaryOperatorKind operatorKind, ExpressionSyntax value, IOperation target) : AnalyzedPattern(target)
    {
        public readonly BinaryOperatorKind OperatorKind = operatorKind;
        public readonly ExpressionSyntax Value = value;
    }
 
    /// <summary>
    /// Represents an and/or pattern, constructed from a logical and/or expression.
    /// </summary>
    internal sealed class Binary : AnalyzedPattern
    {
        public readonly AnalyzedPattern Left;
        public readonly AnalyzedPattern Right;
        public readonly bool IsDisjunctive;
        public readonly SyntaxToken Token;
 
        private Binary(AnalyzedPattern leftPattern, AnalyzedPattern rightPattern, bool isDisjunctive, SyntaxToken token, IOperation target) : base(target)
        {
            Left = leftPattern;
            Right = rightPattern;
            IsDisjunctive = isDisjunctive;
            Token = token;
        }
 
        public static AnalyzedPattern? TryCreate(AnalyzedPattern leftPattern, AnalyzedPattern rightPattern, bool isDisjunctive, SyntaxToken token)
        {
            var leftTarget = leftPattern.Target;
            var rightTarget = rightPattern.Target;
 
            var leftConv = (leftTarget as IConversionOperation)?.Conversion;
            var rightConv = (rightTarget as IConversionOperation)?.Conversion;
 
            var target = (leftConv, rightConv) switch
            {
                ({ IsUserDefined: true }, _) or
                (_, { IsUserDefined: true }) => null,
 
                // If the original targets are implicitly converted due to usage of operators,
                // both targets must have been converted to the same type, otherwise we bail.
                ({ IsImplicit: true }, { IsImplicit: true }) when !Equals(leftTarget.Type, rightTarget.Type) => null,
 
                // If either of targets are implicitly converted but not both,
                // we take the conversion node so that we can generate a cast off of it.
                (null, { IsImplicit: true }) => rightTarget,
                ({ IsImplicit: true }, null) => leftTarget,
 
                // If no implicit conversion is present, we just pick either side and continue.
                _ => leftTarget,
            };
 
            if (target is null)
                return null;
 
            var compareTarget = target == leftTarget ? rightTarget : leftTarget;
            if (!AreEquivalent(target.Syntax, compareTarget.Syntax))
                return null;
 
            // An &&/|| expression that looks the same on both sides.  This looks like something we could merge into an
            // 'and/or' pattern that only executes the target once, and compares the result to multiple constants.
            // However, we disqualify certain forms as they strongly imply that executing the target multiple times
            // is necessary.  For example:
            //
            //  `if (ReadValue(x, ref index) == A && ReadValue(x, ref index) == B)`
            //
            // This should not get converted to `if (ReadValue(x, ref index) == A and B)`.
            //
            // This list is not exhaustive and can be expanded as needed.
            if (target.Syntax.DescendantNodesAndSelf().OfType<ArgumentSyntax>().Any(a => a.RefKindKeyword.Kind() is SyntaxKind.RefKeyword))
                return null;
 
            return new Binary(leftPattern, rightPattern, isDisjunctive, token, target);
        }
    }
 
    /// <summary>
    /// Represents a not-pattern, constructed from inequality check or a logical-not expression.
    /// </summary>
    internal sealed class Not : AnalyzedPattern
    {
        public readonly AnalyzedPattern Pattern;
 
        private Not(AnalyzedPattern pattern, IOperation target) : base(target)
            => Pattern = pattern;
 
        private static BinaryOperatorKind Negate(BinaryOperatorKind kind)
        {
            return kind switch
            {
                BinaryOperatorKind.LessThan => BinaryOperatorKind.GreaterThanOrEqual,
                BinaryOperatorKind.GreaterThan => BinaryOperatorKind.LessThanOrEqual,
                BinaryOperatorKind.LessThanOrEqual => BinaryOperatorKind.GreaterThan,
                BinaryOperatorKind.GreaterThanOrEqual => BinaryOperatorKind.LessThan,
                var v => throw ExceptionUtilities.UnexpectedValue(v)
            };
        }
 
        public static AnalyzedPattern? TryCreate(AnalyzedPattern? pattern)
        {
            return pattern switch
            {
                null => null,
                Not p => p.Pattern, // Avoid double negative
                Relational p => new Relational(Negate(p.OperatorKind), p.Value, p.Target),
                Binary { Left: Not left, Right: Not right } p // Apply demorgans's law
                    => Binary.TryCreate(left.Pattern, right.Pattern, !p.IsDisjunctive, p.Token),
                _ => new Not(pattern, pattern.Target)
            };
        }
    }
}