|
// 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.Collections.Immutable;
using System.Diagnostics;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.PooledObjects;
namespace Microsoft.CodeAnalysis.CSharp.Symbols
{
/// <summary>
/// This method encodes the information needed to round-trip extension types
/// through metadata.
///
/// It encodes receiver parameter - the only parameter of the method
/// </summary>
internal sealed class SynthesizedExtensionMarker : SynthesizedSourceOrdinaryMethodSymbol
{
internal SynthesizedExtensionMarker(SourceMemberContainerTypeSymbol extensionType, ParameterListSyntax parameterList)
: base(extensionType, WellKnownMemberNames.ExtensionMarkerMethodName, parameterList.OpenParenToken.GetLocation(), parameterList,
(GetDeclarationModifiers(), MakeFlags(
MethodKind.Ordinary, RefKind.None, GetDeclarationModifiers(), returnsVoid: false, returnsVoidIsSet: false,
isExpressionBodied: false, isExtensionMethod: false, isNullableAnalysisEnabled: false, isVarArg: false,
isExplicitInterfaceImplementation: false, hasThisInitializer: false)))
{
Debug.Assert(extensionType.IsDefinition);
Debug.Assert(extensionType.IsExtension);
Debug.Assert(parameterList is not null);
return;
}
private static DeclarationModifiers GetDeclarationModifiers() => DeclarationModifiers.Private | DeclarationModifiers.Static;
internal override bool HasSpecialName => true; // Tracked by https://github.com/dotnet/roslyn/issues/76130 : reconcile with spec
internal override void GenerateMethodBody(TypeCompilationState compilationState, BindingDiagnosticBag diagnostics)
{
var F = new SyntheticBoundNodeFactory(this, ContainingType.GetNonNullSyntaxNode(), compilationState, diagnostics);
F.CloseMethod(F.Return());
}
protected override int GetParameterCountFromSyntax()
{
ParameterListSyntax parameterList = (ParameterListSyntax)syntaxReferenceOpt.GetSyntax();
return parameterList.Parameters is [{ IsArgList: false }, ..] ? 1 : 0;
}
protected override (TypeWithAnnotations ReturnType, ImmutableArray<ParameterSymbol> Parameters) MakeParametersAndBindReturnType(BindingDiagnosticBag diagnostics)
{
return (TypeWithAnnotations.Create(Binder.GetSpecialType(DeclaringCompilation, SpecialType.System_Void, GetFirstLocation(), diagnostics)),
makeExtensionParameter(diagnostics) is { } parameter ? [parameter] : []);
ParameterSymbol? makeExtensionParameter(BindingDiagnosticBag diagnostics)
{
ParameterListSyntax parameterList = (ParameterListSyntax)syntaxReferenceOpt.GetSyntax();
int count = parameterList.Parameters.Count;
if (count == 0)
{
return null;
}
BinderFactory binderFactory = this.DeclaringCompilation.GetBinderFactory(parameterList.SyntaxTree);
var withTypeParamsBinder = binderFactory.GetBinder(parameterList);
// Constraints are checked later
var signatureBinder = withTypeParamsBinder.WithAdditionalFlagsAndContainingMemberOrLambda(BinderFlags.SuppressConstraintChecks, this);
for (int parameterIndex = 1; parameterIndex < count; parameterIndex++)
{
diagnostics.Add(ErrorCode.ERR_ReceiverParameterOnlyOne, parameterList.Parameters[parameterIndex].GetLocation());
}
ParameterSymbol? parameter = ParameterHelpers.MakeExtensionReceiverParameter(withTypeParametersBinder: signatureBinder, owner: this, parameterList, diagnostics);
if (parameter is { })
{
checkUnderspecifiedGenericExtension(parameter, ContainingType.TypeParameters, diagnostics);
TypeSymbol parameterType = parameter.TypeWithAnnotations.Type;
RefKind parameterRefKind = parameter.RefKind;
SyntaxNode? parameterTypeSyntax = parameterList.Parameters[0].Type;
Debug.Assert(parameterTypeSyntax is not null);
// Note: SourceOrdinaryMethodSymbol.ExtensionMethodChecks has similar checks, which should be kept in sync.
if (!parameterType.IsValidExtensionParameterType())
{
diagnostics.Add(ErrorCode.ERR_BadTypeforThis, parameterTypeSyntax, parameterType);
}
else if (parameterRefKind == RefKind.Ref && !parameterType.IsValueType)
{
diagnostics.Add(ErrorCode.ERR_RefExtensionParameterMustBeValueTypeOrConstrainedToOne, parameterTypeSyntax);
}
else if (parameterRefKind is RefKind.In or RefKind.RefReadOnlyParameter && parameterType.TypeKind != TypeKind.Struct)
{
diagnostics.Add(ErrorCode.ERR_InExtensionParameterMustBeValueType, parameterTypeSyntax);
}
if (parameter.Name is "" && parameterRefKind != RefKind.None)
{
diagnostics.Add(ErrorCode.ERR_ModifierOnUnnamedReceiverParameter, parameterTypeSyntax);
}
}
if (parameter is { Name: var name } && name != "" &&
ContainingType.TypeParameters.Any(static (p, name) => p.Name == name, name))
{
diagnostics.Add(ErrorCode.ERR_ReceiverParameterSameNameAsTypeParameter, parameter.GetFirstLocation(), name);
}
return parameter;
}
static void checkUnderspecifiedGenericExtension(ParameterSymbol parameter, ImmutableArray<TypeParameterSymbol> typeParameters, BindingDiagnosticBag diagnostics)
{
var underlyingType = parameter.Type;
var usedTypeParameters = PooledHashSet<TypeParameterSymbol>.GetInstance();
underlyingType.VisitType(collectTypeParameters, arg: usedTypeParameters);
foreach (var typeParameter in typeParameters)
{
if (!usedTypeParameters.Contains(typeParameter))
{
diagnostics.Add(ErrorCode.ERR_UnderspecifiedExtension, parameter.GetFirstLocation(), underlyingType, typeParameter);
}
}
usedTypeParameters.Free();
}
static bool collectTypeParameters(TypeSymbol type, PooledHashSet<TypeParameterSymbol> typeParameters, bool ignored)
{
if (type is TypeParameterSymbol typeParameter)
{
typeParameters.Add(typeParameter);
}
return false;
}
}
}
}
|