File: Symbols\Source\SynthesizedMyGroupCollectionPropertyAccessorSymbol.vb
Web Access
Project: src\src\Compilers\VisualBasic\Portable\Microsoft.CodeAnalysis.VisualBasic.vbproj (Microsoft.CodeAnalysis.VisualBasic)
' 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.
 
Imports System.Collections.Generic
Imports System.Collections.Immutable
Imports System.Diagnostics
Imports System.Runtime.InteropServices
Imports System.Text
Imports Microsoft.CodeAnalysis.PooledObjects
Imports Microsoft.CodeAnalysis.Text
Imports Microsoft.CodeAnalysis.VisualBasic.Emit
Imports Microsoft.CodeAnalysis.VisualBasic.Symbols
Imports Microsoft.CodeAnalysis.VisualBasic.Syntax
Imports TypeKind = Microsoft.CodeAnalysis.TypeKind
 
Namespace Microsoft.CodeAnalysis.VisualBasic.Symbols
 
    ''' <summary>
    ''' Represents a compiler "MyGroupCollection" property accessor.
    ''' </summary>
    Friend MustInherit Class SynthesizedMyGroupCollectionPropertyAccessorSymbol
        Inherits SynthesizedPropertyAccessorBase(Of SynthesizedMyGroupCollectionPropertySymbol)
 
        Private ReadOnly _createOrDisposeMethod As String
 
        Public Sub New(container As SourceNamedTypeSymbol, [property] As SynthesizedMyGroupCollectionPropertySymbol, createOrDisposeMethod As String)
            MyBase.New(container, [property])
            Debug.Assert(createOrDisposeMethod IsNot Nothing AndAlso createOrDisposeMethod.Length > 0)
            _createOrDisposeMethod = createOrDisposeMethod
        End Sub
 
        Friend Overrides ReadOnly Property BackingFieldSymbol As FieldSymbol
            Get
                Return PropertyOrEvent.AssociatedField
            End Get
        End Property
 
        Friend Overrides Sub AddSynthesizedAttributes(moduleBuilder As PEModuleBuilder, ByRef attributes As ArrayBuilder(Of VisualBasicAttributeData))
            MyBase.AddSynthesizedAttributes(moduleBuilder, attributes)
 
            ' Note, Dev11 emits DebuggerNonUserCodeAttribute, but we are using DebuggerHiddenAttribute instead.
            AddSynthesizedAttribute(attributes, Me.DeclaringCompilation.SynthesizeDebuggerHiddenAttribute())
        End Sub
 
        Private Shared Function MakeSafeName(name As String) As String
            If SyntaxFacts.GetKeywordKind(name) <> SyntaxKind.None Then
                Return "[" & name & "]"
            End If
 
            Return name
        End Function
 
        Friend Overrides Function GetBoundMethodBody(compilationState As TypeCompilationState, diagnostics As BindingDiagnosticBag, <Out()> Optional ByRef methodBodyBinder As Binder = Nothing) As BoundBlock
 
            Dim containingType = DirectCast(Me.ContainingType, SourceNamedTypeSymbol)
            Dim containingTypeName As String = MakeSafeName(containingType.Name)
 
            Dim targetTypeName As String = PropertyOrEvent.Type.ToDisplayString(SymbolDisplayFormat.FullyQualifiedFormat)
            Debug.Assert(targetTypeName.StartsWith("Global.", StringComparison.Ordinal))
 
            Dim propertyName As String = MakeSafeName(PropertyOrEvent.Name)
 
            Dim fieldName As String = PropertyOrEvent.AssociatedField.Name
 
            Dim codeToParse As String =
                "Partial Class " & containingTypeName & vbCrLf &
                    "Property " & propertyName & vbCrLf &
                        GetMethodBlock(fieldName, MakeSafeName(_createOrDisposeMethod), targetTypeName) &
                    "End Property" & vbCrLf &
                "End Class" & vbCrLf
 
            ' TODO: It looks like Dev11 respects project level conditional compilation here.
            Dim tree = VisualBasicSyntaxTree.ParseText(SourceText.From(codeToParse, Encoding.UTF8, SourceHashAlgorithms.Default))
            Dim attributeSyntax = PropertyOrEvent.AttributeSyntax.GetVisualBasicSyntax()
            Dim diagnosticLocation As Location = attributeSyntax.GetLocation()
            Dim root As CompilationUnitSyntax = tree.GetCompilationUnitRoot()
            Dim hasErrors As Boolean = False
 
            For Each diag As Diagnostic In tree.GetDiagnostics(root)
                Dim vbdiag = DirectCast(diag, VBDiagnostic)
                Debug.Assert(Not vbdiag.HasLazyInfo,
                             "If we decide to allow lazy syntax diagnostics, we'll have to check all call sites of SyntaxTree.GetDiagnostics")
 
                diagnostics.Add(vbdiag.WithLocation(diagnosticLocation))
 
                If diag.Severity = DiagnosticSeverity.Error Then
                    hasErrors = True
                End If
            Next
 
            Dim classBlock = DirectCast(root.Members(0), ClassBlockSyntax)
            Dim propertyBlock = DirectCast(classBlock.Members(0), PropertyBlockSyntax)
            Dim accessorBlock As AccessorBlockSyntax = propertyBlock.Accessors(0)
 
            Dim boundStatement As BoundStatement
 
            If hasErrors Then
                boundStatement = New BoundBadStatement(accessorBlock, ImmutableArray(Of BoundNode).Empty)
            Else
                Dim typeBinder As Binder = BinderBuilder.CreateBinderForType(containingType.ContainingSourceModule, PropertyOrEvent.AttributeSyntax.SyntaxTree, containingType)
                methodBodyBinder = BinderBuilder.CreateBinderForMethodBody(Me, accessorBlock, typeBinder)
 
                Dim bindingDiagnostics = BindingDiagnosticBag.GetInstance(withDiagnostics:=True, withDependencies:=diagnostics.AccumulatesDependencies)
#If DEBUG Then
                ' Enable DEBUG check for ordering of simple name binding.
                methodBodyBinder.EnableSimpleNameBindingOrderChecks(True)
#End If
 
                boundStatement = methodBodyBinder.BindStatement(accessorBlock, bindingDiagnostics)
 
#If DEBUG Then
                methodBodyBinder.EnableSimpleNameBindingOrderChecks(False)
#End If
 
                For Each diag As VBDiagnostic In bindingDiagnostics.DiagnosticBag.AsEnumerable()
                    diagnostics.Add(diag.WithLocation(diagnosticLocation))
                Next
 
                diagnostics.AddDependencies(bindingDiagnostics)
                bindingDiagnostics.Free()
 
                If boundStatement.Kind = BoundKind.Block Then
                    Return DirectCast(boundStatement, BoundBlock)
                End If
            End If
 
            Return New BoundBlock(accessorBlock, Nothing, ImmutableArray(Of LocalSymbol).Empty, ImmutableArray.Create(Of BoundStatement)(boundStatement))
        End Function
 
        Friend NotOverridable Overrides ReadOnly Property GenerateDebugInfoImpl As Boolean
            Get
                Return False
            End Get
        End Property
 
        Friend NotOverridable Overrides Function CalculateLocalSyntaxOffset(localPosition As Integer, localTree As SyntaxTree) As Integer
            Throw ExceptionUtilities.Unreachable
        End Function
 
        Protected MustOverride Function GetMethodBlock(fieldName As String, createOrDisposeMethodName As String, targetTypeName As String) As String
 
    End Class
 
    Friend Class SynthesizedMyGroupCollectionPropertyGetAccessorSymbol
        Inherits SynthesizedMyGroupCollectionPropertyAccessorSymbol
 
        Public Sub New(container As SourceNamedTypeSymbol, [property] As SynthesizedMyGroupCollectionPropertySymbol, createMethod As String)
            MyBase.New(container, [property], createMethod)
        End Sub
 
        Public Overrides ReadOnly Property IsSub As Boolean
            Get
                Return False
            End Get
        End Property
 
        Public Overrides ReadOnly Property MethodKind As MethodKind
            Get
                Return MethodKind.PropertyGet
            End Get
        End Property
 
        Public Overrides ReadOnly Property ReturnType As TypeSymbol
            Get
                Return PropertyOrEvent.Type
            End Get
        End Property
 
        Protected Overrides Function GetMethodBlock(fieldName As String, createMethodName As String, targetTypeName As String) As String
            ' See Bindable::GenMyGroupCollectionGetCode.
 
            ' Get
            '    <backingField> = <CreateMethod>(Of <TargetType>)(<backingField>)
            '    return <backingField>
            ' End Get
 
            Return "Get" & vbCrLf &
                       fieldName & " = " & createMethodName & "(Of " & targetTypeName & ")(" & fieldName & ")" & vbCrLf &
                       "Return " & fieldName & vbCrLf &
                   "End Get" & vbCrLf
        End Function
    End Class
 
    Friend Class SynthesizedMyGroupCollectionPropertySetAccessorSymbol
        Inherits SynthesizedMyGroupCollectionPropertyAccessorSymbol
 
        Private ReadOnly _parameters As ImmutableArray(Of ParameterSymbol)
 
        Public Sub New(container As SourceNamedTypeSymbol, [property] As SynthesizedMyGroupCollectionPropertySymbol, disposeMethod As String)
            MyBase.New(container, [property], disposeMethod)
 
            Dim params() As ParameterSymbol = {SynthesizedParameterSymbol.CreateSetAccessorValueParameter(Me, [property], StringConstants.ValueParameterName)}
            _parameters = params.AsImmutableOrNull()
        End Sub
 
        Public Overrides ReadOnly Property IsSub As Boolean
            Get
                Return True
            End Get
        End Property
 
        Public Overrides ReadOnly Property MethodKind As MethodKind
            Get
                Return MethodKind.PropertySet
            End Get
        End Property
 
        Public Overrides ReadOnly Property ReturnType As TypeSymbol
            Get
                ' There is no reason to specially complain about missing/bad System.Void because we require presence of constructor,
                ' which also returns void. The error reported on the constructor is sufficient.
                Return ContainingAssembly.GetSpecialType(SpecialType.System_Void)
            End Get
        End Property
 
        Public Overrides ReadOnly Property Parameters As ImmutableArray(Of ParameterSymbol)
            Get
                Return _parameters
            End Get
        End Property
 
        Protected Overrides Function GetMethodBlock(fieldName As String, disposeMethodName As String, targetTypeName As String) As String
            ' See Bindable::GenMyGroupCollectionSetCode.
 
            ' Set(ByVal <Value> As <TargetType>)
            '    If <Value> Is <backingField>
            '        return
            '    End If
            '    If Not <Value> Is Nothing Then
            '        Throw New ArgumentException("Property can only be set to Nothing.")
            '    End If
            '    <DisposeMethod>(Of <TargetType>)(<backingField>)
            ' End Set
 
            Return "Set(ByVal " & StringConstants.ValueParameterName & " As " & targetTypeName & ")" & vbCrLf &
                       "If " & StringConstants.ValueParameterName & " Is " & fieldName & vbCrLf &
                           "Return" & vbCrLf &
                       "End If" & vbCrLf &
                       "If " & StringConstants.ValueParameterName & " IsNot Nothing Then" & vbCrLf &
                           "Throw New Global.System.ArgumentException(""Property can only be set to Nothing"")" & vbCrLf &
                       "End If" & vbCrLf &
                       disposeMethodName & "(Of " & targetTypeName & ")(" & fieldName & ")" & vbCrLf &
                   "End Set" & vbCrLf
        End Function
    End Class
 
End Namespace