From 9c315035850b619550b30c50c9521c16a2cbb5f6 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 26 Jan 2018 03:29:32 +0330 Subject: [PATCH 01/43] Implement stackalloc initializers --- .../Portable/Binder/BinderFlagsExtensions.cs | 5 + .../Portable/Binder/Binder_Conversions.cs | 2 +- .../Portable/Binder/Binder_Expressions.cs | 160 ++- .../CSharp/Portable/BoundTree/BoundNodes.xml | 3 +- .../Portable/CSharpResources.Designer.cs | 9 + .../CSharp/Portable/CSharpResources.resx | 3 + .../CSharp/Portable/CodeGen/EmitExpression.cs | 17 + .../CodeGen/EmitStackAllocInitializer.cs | 153 +++ .../CSharp/Portable/Errors/MessageID.cs | 3 + .../Generated/BoundNodes.xml.Generated.cs | 36 +- .../Syntax.xml.Internal.Generated.cs | 320 ++++- .../Generated/Syntax.xml.Main.Generated.cs | 72 +- .../Generated/Syntax.xml.Syntax.Generated.cs | 131 +- .../LocalRewriter_PointerElementAccess.cs | 7 +- .../LocalRewriter/LocalRewriter_StackAlloc.cs | 10 +- .../CSharp/Portable/Parser/LanguageParser.cs | 52 +- .../CSharp/Portable/PublicAPI.Unshipped.txt | 26 +- ...StackAllocArrayCreationExpressionSyntax.cs | 25 + .../CSharp/Portable/Syntax/Syntax.xml | 37 + .../CSharp/Portable/Syntax/SyntaxKind.cs | 2 + .../CodeGen/StackAllocInitializerTests.cs | 388 ++++++ .../Semantics/StackAllocInitializerTests.cs | 1137 +++++++++++++++++ .../Generated/Syntax.Test.xml.Generated.cs | 96 +- .../Syntax/Parsing/ExpressionParsingTests.cs | 1 + .../Test/Syntax/Parsing/ParsingTests.cs | 9 +- .../Parsing/StackAllocInitializerTests.cs | 113 ++ .../Core/Portable/CodeGen/ILBuilderEmit.cs | 24 + src/Test/Utilities/Portable/TestResource.resx | 2 +- .../Portable/Traits/CompilerFeature.cs | 1 + 29 files changed, 2755 insertions(+), 89 deletions(-) create mode 100644 src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs create mode 100644 src/Compilers/CSharp/Portable/Syntax/StackAllocArrayCreationExpressionSyntax.cs create mode 100644 src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs create mode 100644 src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs create mode 100644 src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs diff --git a/src/Compilers/CSharp/Portable/Binder/BinderFlagsExtensions.cs b/src/Compilers/CSharp/Portable/Binder/BinderFlagsExtensions.cs index e437c80e8e535..144de3cc392ec 100644 --- a/src/Compilers/CSharp/Portable/Binder/BinderFlagsExtensions.cs +++ b/src/Compilers/CSharp/Portable/Binder/BinderFlagsExtensions.cs @@ -11,5 +11,10 @@ public static bool Includes(this BinderFlags self, BinderFlags other) { return (self & other) == other; } + + public static bool IncludesAny(this BinderFlags self, BinderFlags other) + { + return (self & other) != 0; + } } } diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Conversions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Conversions.cs index 7bc3346493e5b..87c47dc836c8e 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Conversions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Conversions.cs @@ -339,7 +339,7 @@ private BoundExpression CreateStackAllocConversion(SyntaxNode syntax, BoundExpre throw ExceptionUtilities.UnexpectedValue(conversion.Kind); } - var convertedNode = new BoundConvertedStackAllocExpression(syntax, elementType, boundStackAlloc.Count, stackAllocType, boundStackAlloc.HasErrors); + var convertedNode = new BoundConvertedStackAllocExpression(syntax, elementType, boundStackAlloc.Count, boundStackAlloc.InitializerOpt, stackAllocType, boundStackAlloc.HasErrors); var underlyingConversion = conversion.UnderlyingConversions.Single(); return CreateConversion(syntax, convertedNode, underlyingConversion, isCast, destination, diagnostics); diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index b5e2811d66d11..cff5dc0a2abdb 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -383,6 +383,8 @@ private BoundExpression BindExpressionInternal(ExpressionSyntax node, Diagnostic return BindImplicitArrayCreationExpression((ImplicitArrayCreationExpressionSyntax)node, diagnostics); case SyntaxKind.StackAllocArrayCreationExpression: return BindStackAllocArrayCreationExpression((StackAllocArrayCreationExpressionSyntax)node, diagnostics); + case SyntaxKind.ImplicitStackAllocArrayCreationExpression: + return BindImplicitStackAllocArrayCreationExpression((ImplicitStackAllocArrayCreationExpressionSyntax)node, diagnostics); case SyntaxKind.ObjectCreationExpression: return BindObjectCreationExpression((ObjectCreationExpressionSyntax)node, diagnostics); case SyntaxKind.IdentifierName: @@ -2714,6 +2716,40 @@ private BoundExpression BindImplicitArrayCreationExpression(ImplicitArrayCreatio sizes: ImmutableArray.Empty, boundInitExprOpt: boundInitializerExpressions); } + + private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node, DiagnosticBag diagnostics) + { + bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); + bool hasErrors = !inLegalPosition; + + InitializerExpressionSyntax initializer = node.Initializer; + ImmutableArray boundInitializerExpressions = BindArrayInitializerExpressions(initializer, diagnostics, dimension: 1, rank: 1); + + HashSet useSiteDiagnostics = null; + TypeSymbol bestType = BestTypeInferrer.InferBestType(boundInitializerExpressions, this.Conversions, out bool hadMultipleCandidates, ref useSiteDiagnostics); + diagnostics.Add(node, useSiteDiagnostics); + + if ((object)bestType == null || bestType.SpecialType == SpecialType.System_Void) + { + Error(diagnostics, ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, node); + bestType = CreateErrorType(); + } + + if (!bestType.IsErrorType() && bestType.IsManagedType) + { + Error(diagnostics, ErrorCode.ERR_ManagedAddr, node, bestType); + } + + return BindStackAllocWithInitializer( + node, + initializer, + type: GetStackAllocType(node, bestType, inLegalPosition, diagnostics), + elementType: bestType, + sizeOpt: null, + diagnostics, + hasErrors: hasErrors); + } + // This method binds all the array initializer expressions. // NOTE: It doesn't convert the bound initializer expressions to array's element type. // NOTE: This is done separately in ConvertAndBindArrayInitialization method below. @@ -3032,22 +3068,11 @@ private BoundArrayCreation BindArrayCreationWithInitializer( private BoundExpression BindStackAllocArrayCreationExpression( StackAllocArrayCreationExpressionSyntax node, DiagnosticBag diagnostics) { - bool hasErrors = false; - var inLegalPosition = (IsInMethodBody || IsLocalFunctionsScopeBinder) && node.IsLegalSpanStackAllocPosition(); - - if (!inLegalPosition) - { - hasErrors = true; - diagnostics.Add( - ErrorCode.ERR_InvalidExprTerm, - node.StackAllocKeyword.GetLocation(), - SyntaxFacts.GetText(SyntaxKind.StackAllocKeyword)); - } + bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); + bool hasErrors = !inLegalPosition; // Check if we're syntactically within a catch or finally clause. - if (this.Flags.Includes(BinderFlags.InCatchBlock) || - this.Flags.Includes(BinderFlags.InCatchFilter) || - this.Flags.Includes(BinderFlags.InFinallyBlock)) + if (this.Flags.IncludesAny(BinderFlags.InCatchBlock | BinderFlags.InCatchFilter | BinderFlags.InFinallyBlock)) { Error(diagnostics, ErrorCode.ERR_StackallocInCatchFinally, node); } @@ -3070,24 +3095,22 @@ private BoundExpression BindStackAllocArrayCreationExpression( TypeSyntax elementTypeSyntax = arrayTypeSyntax.ElementType; TypeSymbol elementType = BindType(elementTypeSyntax, diagnostics); - bool typeHasErrors = elementType.IsErrorType(); - if (!typeHasErrors && elementType.IsManagedType) + if (!elementType.IsErrorType() && elementType.IsManagedType) { Error(diagnostics, ErrorCode.ERR_ManagedAddr, elementTypeSyntax, elementType); - typeHasErrors = true; + hasErrors = true; } SyntaxList rankSpecifiers = arrayTypeSyntax.RankSpecifiers; if (rankSpecifiers.Count != 1 || - rankSpecifiers[0].Sizes.Count != 1 || - rankSpecifiers[0].Sizes[0].Kind() == SyntaxKind.OmittedArraySizeExpression) + rankSpecifiers[0].Sizes.Count != 1) { // NOTE: Dev10 reported several parse errors here. Error(diagnostics, ErrorCode.ERR_BadStackAllocExpr, typeSyntax); var builder = ArrayBuilder.GetInstance(); - DiagnosticBag discardedDiagnostics = DiagnosticBag.GetInstance(); + var discardedDiagnostics = DiagnosticBag.GetInstance(); foreach (ArrayRankSpecifierSyntax rankSpecifier in rankSpecifiers) { foreach (ExpressionSyntax size in rankSpecifier.Sizes) @@ -3098,6 +3121,7 @@ private BoundExpression BindStackAllocArrayCreationExpression( } } } + discardedDiagnostics.Free(); return new BoundBadExpression( @@ -3108,20 +3132,54 @@ private BoundExpression BindStackAllocArrayCreationExpression( new PointerTypeSymbol(elementType)); } + TypeSymbol type = GetStackAllocType(node, elementType, inLegalPosition, diagnostics); + ExpressionSyntax countSyntax = rankSpecifiers[0].Sizes[0]; - var count = BindValue(countSyntax, diagnostics, BindValueKind.RValue); - if (!count.HasAnyErrors) + BoundExpression count = null; + if (countSyntax.Kind() != SyntaxKind.OmittedArraySizeExpression) { - // NOTE: this is different from how we would bind an array size (in which case we would allow uint, long, or ulong). - count = GenerateConversionForAssignment(GetSpecialType(SpecialType.System_Int32, diagnostics, node), count, diagnostics); - if (!count.HasAnyErrors && IsNegativeConstantForArraySize(count)) + count = BindValue(countSyntax, diagnostics, BindValueKind.RValue); + if (!count.HasAnyErrors) { - Error(diagnostics, ErrorCode.ERR_NegativeStackAllocSize, countSyntax); - hasErrors = true; + // NOTE: this is different from how we would bind an array size (in which case we would allow uint, long, or ulong). + count = GenerateConversionForAssignment(GetSpecialType(SpecialType.System_Int32, diagnostics, node), count, diagnostics); + if (!count.HasAnyErrors && IsNegativeConstantForArraySize(count)) + { + Error(diagnostics, ErrorCode.ERR_NegativeStackAllocSize, countSyntax); + hasErrors = true; + } } } + else if (node.Initializer == null) + { + // ERR_MissingArraySize is already reported + count = BadExpression(countSyntax); + hasErrors = true; + } - TypeSymbol type = null; + return node.Initializer == null + ? new BoundStackAllocArrayCreation(node, elementType, count, initializerOpt: null, type, hasErrors: hasErrors) + : BindStackAllocWithInitializer(node, node.Initializer, type, elementType, count, diagnostics, hasErrors); + } + + private bool ReportBadStackAllocPosition(SyntaxNode node, DiagnosticBag diagnostics) + { + Debug.Assert(node is StackAllocArrayCreationExpressionSyntax || node is ImplicitStackAllocArrayCreationExpressionSyntax); + + var inLegalPosition = (IsInMethodBody || IsLocalFunctionsScopeBinder) && node.IsLegalSpanStackAllocPosition(); + if (!inLegalPosition) + { + diagnostics.Add( + ErrorCode.ERR_InvalidExprTerm, + node.GetFirstToken().GetLocation(), + SyntaxFacts.GetText(SyntaxKind.StackAllocKeyword)); + } + + return inLegalPosition; + } + + private TypeSymbol GetStackAllocType(SyntaxNode node, TypeSymbol elementType, bool inLegalPosition, DiagnosticBag diagnostics) + { if (inLegalPosition && !node.IsVariableDeclarationInitialization()) { CheckFeatureAvailability(node, MessageID.IDS_FeatureRefStructs, diagnostics); @@ -3130,11 +3188,53 @@ private BoundExpression BindStackAllocArrayCreationExpression( var spanType = GetWellKnownType(WellKnownType.System_Span_T, diagnostics, node); if (!spanType.IsErrorType()) { - type = spanType.Construct(elementType); + return spanType.Construct(elementType); } } - return new BoundStackAllocArrayCreation(node, elementType, count, type, hasErrors: hasErrors || typeHasErrors); + return null; + } + + private BoundExpression BindStackAllocWithInitializer( + SyntaxNode node, + InitializerExpressionSyntax initSyntax, + TypeSymbol type, + TypeSymbol elementType, + BoundExpression sizeOpt, + DiagnosticBag diagnostics, + bool hasErrors, + ImmutableArray boundInitExprOpt = default) + { + + if (boundInitExprOpt.IsDefault) + { + boundInitExprOpt = BindArrayInitializerExpressions(initSyntax, diagnostics, dimension: 1, rank: 1) + .SelectAsArray((expr, t) => GenerateConversionForAssignment(t.elementType, expr, t.diagnostics), (elementType, diagnostics)); + } + + if (sizeOpt != null) + { + int? constantSizeOpt = GetIntegerConstantForArraySize(sizeOpt); + if (!sizeOpt.HasAnyErrors && constantSizeOpt == null) + { + Error(diagnostics, ErrorCode.ERR_ConstantExpected, sizeOpt.Syntax); + hasErrors = true; + } + else if (boundInitExprOpt.Length != constantSizeOpt) + { + Error(diagnostics, ErrorCode.ERR_ArrayInitializerIncorrectLength, node, constantSizeOpt.Value); + hasErrors = true; + } + } + else + { + sizeOpt = new BoundLiteral( + node, + ConstantValue.Create(boundInitExprOpt.Length), + GetSpecialType(SpecialType.System_Int32, diagnostics, node)) + { WasCompilerGenerated = true }; + } + return new BoundStackAllocArrayCreation(node, elementType, sizeOpt, new BoundArrayInitialization(initSyntax, boundInitExprOpt), type, hasErrors); } private static int? GetIntegerConstantForArraySize(BoundExpression expression) diff --git a/src/Compilers/CSharp/Portable/BoundTree/BoundNodes.xml b/src/Compilers/CSharp/Portable/BoundTree/BoundNodes.xml index 560e88a72f770..d5e19103b1a14 100644 --- a/src/Compilers/CSharp/Portable/BoundTree/BoundNodes.xml +++ b/src/Compilers/CSharp/Portable/BoundTree/BoundNodes.xml @@ -1444,7 +1444,8 @@ - + + diff --git a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs index 4f5b1d799f80c..8bceed68e9662 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs +++ b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs @@ -10745,6 +10745,15 @@ internal static string IDS_FeatureRefStructs { } } + /// + /// Looks up a localized string similar to stackalloc initilizer. + /// + internal static string IDS_FeatureStackAllocInitializer { + get { + return ResourceManager.GetString("IDS_FeatureStackAllocInitializer", resourceCulture); + } + } + /// /// Looks up a localized string similar to static classes. /// diff --git a/src/Compilers/CSharp/Portable/CSharpResources.resx b/src/Compilers/CSharp/Portable/CSharpResources.resx index 3178e2140048f..be6f120074e14 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.resx +++ b/src/Compilers/CSharp/Portable/CSharpResources.resx @@ -5253,4 +5253,7 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Arguments with 'in' modifier cannot be used in dynamically dispatched expessions. + + stackalloc initilizer + diff --git a/src/Compilers/CSharp/Portable/CodeGen/EmitExpression.cs b/src/Compilers/CSharp/Portable/CodeGen/EmitExpression.cs index 01dc17ea262b3..5e9885e1b3f35 100644 --- a/src/Compilers/CSharp/Portable/CodeGen/EmitExpression.cs +++ b/src/Compilers/CSharp/Portable/CodeGen/EmitExpression.cs @@ -1889,6 +1889,23 @@ private void EmitConvertedStackAllocExpression(BoundConvertedStackAllocExpressio { _builder.EmitOpCode(ILOpCode.Localloc); } + + var initializer = expression.InitializerOpt; + if (initializer != null) + { + if (used) + { + EmitStackAllocInitializers(expression.Type, initializer); + } + else + { + // If not used, just emit initializer elements to preserve possible sideeffects + foreach (var init in initializer.Initializers) + { + EmitExpression(init, used: false); + } + } + } } private void EmitObjectCreationExpression(BoundObjectCreationExpression expression, bool used) diff --git a/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs new file mode 100644 index 0000000000000..73acd341f9123 --- /dev/null +++ b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs @@ -0,0 +1,153 @@ +// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using System; +using System.Collections.Immutable; +using System.Diagnostics; +using System.Linq; +using System.Reflection.Metadata; +using Microsoft.CodeAnalysis.CSharp.Symbols; +using Microsoft.CodeAnalysis.PooledObjects; + +namespace Microsoft.CodeAnalysis.CSharp.CodeGen +{ + internal partial class CodeGenerator + { + private void EmitStackAllocInitializers(TypeSymbol type, BoundArrayInitialization inits) + { + Debug.Assert(type is PointerTypeSymbol || type is NamedTypeSymbol); + + var elementType = type.TypeKind == TypeKind.Pointer + ? ((PointerTypeSymbol)type).PointedAtType + : ((NamedTypeSymbol)type).TypeArgumentsNoUseSiteDiagnostics[0]; + + var initExprs = inits.Initializers; + + var initializationStyle = ShouldEmitBlockInitializerForStackAlloc(elementType, initExprs); + if (initializationStyle == ArrayInitializerStyle.Element) + { + EmitElementStackAllocInitializers(elementType, initExprs, includeConstants: true); + } + else + { + ImmutableArray data = this.GetRawData(initExprs); + _builder.EmitStackAllocBlockInitializer(data, inits.Syntax, _diagnostics); + + if (initializationStyle == ArrayInitializerStyle.Mixed) + { + EmitElementStackAllocInitializers(elementType, initExprs, includeConstants: false); + } + } + } + + private ArrayInitializerStyle ShouldEmitBlockInitializerForStackAlloc(TypeSymbol elementType, ImmutableArray inits) + { + if (!_module.SupportsPrivateImplClass) + { + return ArrayInitializerStyle.Element; + } + + elementType = elementType.EnumUnderlyingType(); + + if (elementType.SpecialType.IsBlittable()) + { + int initCount = 0; + int constCount = 0; + StackAllocInitializerCount(inits, ref initCount, ref constCount); + + if (initCount > 2) + { + if (initCount == constCount) + { + return ArrayInitializerStyle.Block; + } + + int thresholdCnt = Math.Max(3, (initCount / 3)); + + if (constCount >= thresholdCnt) + { + return ArrayInitializerStyle.Mixed; + } + } + } + + return ArrayInitializerStyle.Element; + } + + private void StackAllocInitializerCount(ImmutableArray inits, ref int initCount, ref int constInits) + { + if (inits.Length == 0) + { + return; + } + + foreach (var init in inits) + { + Debug.Assert(!(init is BoundArrayInitialization), "Nested initializers are not allowed for stackalloc"); + + initCount += 1; + if (init.ConstantValue != null) + { + constInits += 1; + } + } + } + + private void EmitElementStackAllocInitializers(TypeSymbol elementType, ImmutableArray inits, bool includeConstants) + { + int index = 0; + int elementTypeSizeInBytes = elementType.SpecialType.SizeInBytes(); + foreach (BoundExpression init in inits) + { + if (includeConstants || init.ConstantValue == null) + { + _builder.EmitOpCode(ILOpCode.Dup); + EmitPointerElementAccess(init, elementType, elementTypeSizeInBytes, index); + EmitExpression(init, used: true); + EmitIndirectStore(elementType, init.Syntax); + } + + index++; + } + } + + private void EmitPointerElementAccess(BoundExpression init, TypeSymbol elementType, int elementTypeSizeInBytes, int index) + { + if (index == 0) + { + return; + } + + if (elementTypeSizeInBytes == 1) + { + _builder.EmitIntConstant(index); + _builder.EmitOpCode(ILOpCode.Add); + } + else if (index == 1) + { + EmitIntConstantOrSizeOf(init, elementType, elementTypeSizeInBytes); + _builder.EmitOpCode(ILOpCode.Add); + } + else + { + _builder.EmitIntConstant(index); + _builder.EmitOpCode(ILOpCode.Conv_i); + EmitIntConstantOrSizeOf(init, elementType, elementTypeSizeInBytes); + _builder.EmitOpCode(ILOpCode.Mul); + _builder.EmitOpCode(ILOpCode.Add); + } + } + + private void EmitIntConstantOrSizeOf(BoundExpression init, TypeSymbol elementType, int elementTypeSizeInBytes) + { + if (elementTypeSizeInBytes == 0) + { + _builder.EmitOpCode(ILOpCode.Sizeof); + EmitSymbolToken(elementType, init.Syntax); + } + else + { + _builder.EmitIntConstant(elementTypeSizeInBytes); + } + } + } +} diff --git a/src/Compilers/CSharp/Portable/Errors/MessageID.cs b/src/Compilers/CSharp/Portable/Errors/MessageID.cs index 3cc8de8a2d211..30883e594ee18 100644 --- a/src/Compilers/CSharp/Portable/Errors/MessageID.cs +++ b/src/Compilers/CSharp/Portable/Errors/MessageID.cs @@ -148,6 +148,8 @@ internal enum MessageID IDS_FeatureRefConditional = MessageBase + 12731, IDS_FeatureAttributesOnBackingFields = MessageBase + 12732, + + IDS_FeatureStackAllocInitializer = MessageBase + 12733, } // Message IDs may refer to strings that need to be localized. @@ -189,6 +191,7 @@ internal static LanguageVersion RequiredVersion(this MessageID feature) { // C# 7.3 features. case MessageID.IDS_FeatureAttributesOnBackingFields: // semantic check + case MessageID.IDS_FeatureStackAllocInitializer: return LanguageVersion.CSharp7_3; // C# 7.2 features. diff --git a/src/Compilers/CSharp/Portable/Generated/BoundNodes.xml.Generated.cs b/src/Compilers/CSharp/Portable/Generated/BoundNodes.xml.Generated.cs index e3dad0c56ab48..76063d9131f70 100644 --- a/src/Compilers/CSharp/Portable/Generated/BoundNodes.xml.Generated.cs +++ b/src/Compilers/CSharp/Portable/Generated/BoundNodes.xml.Generated.cs @@ -5246,7 +5246,7 @@ public BoundArrayInitialization Update(ImmutableArray initializ internal partial class BoundStackAllocArrayCreation : BoundExpression { - protected BoundStackAllocArrayCreation(BoundKind kind, SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, TypeSymbol type, bool hasErrors = false) + protected BoundStackAllocArrayCreation(BoundKind kind, SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, BoundArrayInitialization initializerOpt, TypeSymbol type, bool hasErrors = false) : base(kind, syntax, type, hasErrors) { @@ -5255,10 +5255,11 @@ protected BoundStackAllocArrayCreation(BoundKind kind, SyntaxNode syntax, TypeSy this.ElementType = elementType; this.Count = count; + this.InitializerOpt = initializerOpt; } - public BoundStackAllocArrayCreation(SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, TypeSymbol type, bool hasErrors = false) - : base(BoundKind.StackAllocArrayCreation, syntax, type, hasErrors || count.HasErrors()) + public BoundStackAllocArrayCreation(SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, BoundArrayInitialization initializerOpt, TypeSymbol type, bool hasErrors = false) + : base(BoundKind.StackAllocArrayCreation, syntax, type, hasErrors || count.HasErrors() || initializerOpt.HasErrors()) { Debug.Assert(elementType != null, "Field 'elementType' cannot be null (use Null=\"allow\" in BoundNodes.xml to remove this check)"); @@ -5266,6 +5267,7 @@ public BoundStackAllocArrayCreation(SyntaxNode syntax, TypeSymbol elementType, B this.ElementType = elementType; this.Count = count; + this.InitializerOpt = initializerOpt; } @@ -5273,16 +5275,18 @@ public BoundStackAllocArrayCreation(SyntaxNode syntax, TypeSymbol elementType, B public BoundExpression Count { get; } + public BoundArrayInitialization InitializerOpt { get; } + public override BoundNode Accept(BoundTreeVisitor visitor) { return visitor.VisitStackAllocArrayCreation(this); } - public BoundStackAllocArrayCreation Update(TypeSymbol elementType, BoundExpression count, TypeSymbol type) + public BoundStackAllocArrayCreation Update(TypeSymbol elementType, BoundExpression count, BoundArrayInitialization initializerOpt, TypeSymbol type) { - if (elementType != this.ElementType || count != this.Count || type != this.Type) + if (elementType != this.ElementType || count != this.Count || initializerOpt != this.InitializerOpt || type != this.Type) { - var result = new BoundStackAllocArrayCreation(this.Syntax, elementType, count, type, this.HasErrors); + var result = new BoundStackAllocArrayCreation(this.Syntax, elementType, count, initializerOpt, type, this.HasErrors); result.WasCompilerGenerated = this.WasCompilerGenerated; return result; } @@ -5292,8 +5296,8 @@ public BoundStackAllocArrayCreation Update(TypeSymbol elementType, BoundExpressi internal sealed partial class BoundConvertedStackAllocExpression : BoundStackAllocArrayCreation { - public BoundConvertedStackAllocExpression(SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, TypeSymbol type, bool hasErrors = false) - : base(BoundKind.ConvertedStackAllocExpression, syntax, elementType, count, type, hasErrors || count.HasErrors()) + public BoundConvertedStackAllocExpression(SyntaxNode syntax, TypeSymbol elementType, BoundExpression count, BoundArrayInitialization initializerOpt, TypeSymbol type, bool hasErrors = false) + : base(BoundKind.ConvertedStackAllocExpression, syntax, elementType, count, initializerOpt, type, hasErrors || count.HasErrors() || initializerOpt.HasErrors()) { Debug.Assert(elementType != null, "Field 'elementType' cannot be null (use Null=\"allow\" in BoundNodes.xml to remove this check)"); @@ -5308,11 +5312,11 @@ public override BoundNode Accept(BoundTreeVisitor visitor) return visitor.VisitConvertedStackAllocExpression(this); } - public new BoundConvertedStackAllocExpression Update(TypeSymbol elementType, BoundExpression count, TypeSymbol type) + public new BoundConvertedStackAllocExpression Update(TypeSymbol elementType, BoundExpression count, BoundArrayInitialization initializerOpt, TypeSymbol type) { - if (elementType != this.ElementType || count != this.Count || type != this.Type) + if (elementType != this.ElementType || count != this.Count || initializerOpt != this.InitializerOpt || type != this.Type) { - var result = new BoundConvertedStackAllocExpression(this.Syntax, elementType, count, type, this.HasErrors); + var result = new BoundConvertedStackAllocExpression(this.Syntax, elementType, count, initializerOpt, type, this.HasErrors); result.WasCompilerGenerated = this.WasCompilerGenerated; return result; } @@ -8327,11 +8331,13 @@ public override BoundNode VisitArrayInitialization(BoundArrayInitialization node public override BoundNode VisitStackAllocArrayCreation(BoundStackAllocArrayCreation node) { this.Visit(node.Count); + this.Visit(node.InitializerOpt); return null; } public override BoundNode VisitConvertedStackAllocExpression(BoundConvertedStackAllocExpression node) { this.Visit(node.Count); + this.Visit(node.InitializerOpt); return null; } public override BoundNode VisitFieldAccess(BoundFieldAccess node) @@ -9190,16 +9196,18 @@ public override BoundNode VisitArrayInitialization(BoundArrayInitialization node public override BoundNode VisitStackAllocArrayCreation(BoundStackAllocArrayCreation node) { BoundExpression count = (BoundExpression)this.Visit(node.Count); + BoundArrayInitialization initializerOpt = (BoundArrayInitialization)this.Visit(node.InitializerOpt); TypeSymbol elementType = this.VisitType(node.ElementType); TypeSymbol type = this.VisitType(node.Type); - return node.Update(elementType, count, type); + return node.Update(elementType, count, initializerOpt, type); } public override BoundNode VisitConvertedStackAllocExpression(BoundConvertedStackAllocExpression node) { BoundExpression count = (BoundExpression)this.Visit(node.Count); + BoundArrayInitialization initializerOpt = (BoundArrayInitialization)this.Visit(node.InitializerOpt); TypeSymbol elementType = this.VisitType(node.ElementType); TypeSymbol type = this.VisitType(node.Type); - return node.Update(elementType, count, type); + return node.Update(elementType, count, initializerOpt, type); } public override BoundNode VisitFieldAccess(BoundFieldAccess node) { @@ -10656,6 +10664,7 @@ public override TreeDumperNode VisitStackAllocArrayCreation(BoundStackAllocArray { new TreeDumperNode("elementType", node.ElementType, null), new TreeDumperNode("count", null, new TreeDumperNode[] { Visit(node.Count, null) }), + new TreeDumperNode("initializerOpt", null, new TreeDumperNode[] { Visit(node.InitializerOpt, null) }), new TreeDumperNode("type", node.Type, null) } ); @@ -10666,6 +10675,7 @@ public override TreeDumperNode VisitConvertedStackAllocExpression(BoundConverted { new TreeDumperNode("elementType", node.ElementType, null), new TreeDumperNode("count", null, new TreeDumperNode[] { Visit(node.Count, null) }), + new TreeDumperNode("initializerOpt", null, new TreeDumperNode[] { Visit(node.InitializerOpt, null) }), new TreeDumperNode("type", node.Type, null) } ); diff --git a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Internal.Generated.cs b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Internal.Generated.cs index 2612aff81df34..a6938d53ed7ac 100644 --- a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Internal.Generated.cs +++ b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Internal.Generated.cs @@ -8076,44 +8076,62 @@ internal sealed partial class StackAllocArrayCreationExpressionSyntax : Expressi { internal readonly SyntaxToken stackAllocKeyword; internal readonly TypeSyntax type; + internal readonly InitializerExpressionSyntax initializer; - internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type, DiagnosticInfo[] diagnostics, SyntaxAnnotation[] annotations) + internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer, DiagnosticInfo[] diagnostics, SyntaxAnnotation[] annotations) : base(kind, diagnostics, annotations) { - this.SlotCount = 2; + this.SlotCount = 3; this.AdjustFlagsAndWidth(stackAllocKeyword); this.stackAllocKeyword = stackAllocKeyword; this.AdjustFlagsAndWidth(type); this.type = type; + if (initializer != null) + { + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } } - internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type, SyntaxFactoryContext context) + internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer, SyntaxFactoryContext context) : base(kind) { this.SetFactoryContext(context); - this.SlotCount = 2; + this.SlotCount = 3; this.AdjustFlagsAndWidth(stackAllocKeyword); this.stackAllocKeyword = stackAllocKeyword; this.AdjustFlagsAndWidth(type); this.type = type; + if (initializer != null) + { + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } } - internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type) + internal StackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) : base(kind) { - this.SlotCount = 2; + this.SlotCount = 3; this.AdjustFlagsAndWidth(stackAllocKeyword); this.stackAllocKeyword = stackAllocKeyword; this.AdjustFlagsAndWidth(type); this.type = type; + if (initializer != null) + { + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } } /// SyntaxToken representing the stackalloc keyword. public SyntaxToken StackAllocKeyword { get { return this.stackAllocKeyword; } } /// TypeSyntax node representing the type of the stackalloc array. public TypeSyntax Type { get { return this.type; } } + /// InitializerExpressionSyntax node representing the initializer of the stackalloc array creation expression. + public InitializerExpressionSyntax Initializer { get { return this.initializer; } } internal override GreenNode GetSlot(int index) { @@ -8121,6 +8139,7 @@ internal override GreenNode GetSlot(int index) { case 0: return this.stackAllocKeyword; case 1: return this.type; + case 2: return this.initializer; default: return null; } } @@ -8140,11 +8159,11 @@ public override void Accept(CSharpSyntaxVisitor visitor) visitor.VisitStackAllocArrayCreationExpression(this); } - public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, TypeSyntax type) + public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) { - if (stackAllocKeyword != this.StackAllocKeyword || type != this.Type) + if (stackAllocKeyword != this.StackAllocKeyword || type != this.Type || initializer != this.Initializer) { - var newNode = SyntaxFactory.StackAllocArrayCreationExpression(stackAllocKeyword, type); + var newNode = SyntaxFactory.StackAllocArrayCreationExpression(stackAllocKeyword, type, initializer); var diags = this.GetDiagnostics(); if (diags != null && diags.Length > 0) newNode = newNode.WithDiagnosticsGreen(diags); @@ -8159,18 +8178,18 @@ public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyw internal override GreenNode SetDiagnostics(DiagnosticInfo[] diagnostics) { - return new StackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.type, diagnostics, GetAnnotations()); + return new StackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.type, this.initializer, diagnostics, GetAnnotations()); } internal override GreenNode SetAnnotations(SyntaxAnnotation[] annotations) { - return new StackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.type, GetDiagnostics(), annotations); + return new StackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.type, this.initializer, GetDiagnostics(), annotations); } internal StackAllocArrayCreationExpressionSyntax(ObjectReader reader) : base(reader) { - this.SlotCount = 2; + this.SlotCount = 3; var stackAllocKeyword = (SyntaxToken)reader.ReadValue(); if (stackAllocKeyword != null) { @@ -8183,6 +8202,12 @@ internal StackAllocArrayCreationExpressionSyntax(ObjectReader reader) AdjustFlagsAndWidth(type); this.type = type; } + var initializer = (InitializerExpressionSyntax)reader.ReadValue(); + if (initializer != null) + { + AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } } internal override void WriteTo(ObjectWriter writer) @@ -8190,6 +8215,7 @@ internal override void WriteTo(ObjectWriter writer) base.WriteTo(writer); writer.WriteValue(this.stackAllocKeyword); writer.WriteValue(this.type); + writer.WriteValue(this.initializer); } static StackAllocArrayCreationExpressionSyntax() @@ -8198,6 +8224,167 @@ static StackAllocArrayCreationExpressionSyntax() } } + /// Class which represents the syntax node for implicit stackalloc array creation expression. + internal sealed partial class ImplicitStackAllocArrayCreationExpressionSyntax : ExpressionSyntax + { + internal readonly SyntaxToken stackAllocKeyword; + internal readonly SyntaxToken openBracketToken; + internal readonly SyntaxToken closeBracketToken; + internal readonly InitializerExpressionSyntax initializer; + + internal ImplicitStackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer, DiagnosticInfo[] diagnostics, SyntaxAnnotation[] annotations) + : base(kind, diagnostics, annotations) + { + this.SlotCount = 4; + this.AdjustFlagsAndWidth(stackAllocKeyword); + this.stackAllocKeyword = stackAllocKeyword; + this.AdjustFlagsAndWidth(openBracketToken); + this.openBracketToken = openBracketToken; + this.AdjustFlagsAndWidth(closeBracketToken); + this.closeBracketToken = closeBracketToken; + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } + + + internal ImplicitStackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer, SyntaxFactoryContext context) + : base(kind) + { + this.SetFactoryContext(context); + this.SlotCount = 4; + this.AdjustFlagsAndWidth(stackAllocKeyword); + this.stackAllocKeyword = stackAllocKeyword; + this.AdjustFlagsAndWidth(openBracketToken); + this.openBracketToken = openBracketToken; + this.AdjustFlagsAndWidth(closeBracketToken); + this.closeBracketToken = closeBracketToken; + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } + + + internal ImplicitStackAllocArrayCreationExpressionSyntax(SyntaxKind kind, SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + : base(kind) + { + this.SlotCount = 4; + this.AdjustFlagsAndWidth(stackAllocKeyword); + this.stackAllocKeyword = stackAllocKeyword; + this.AdjustFlagsAndWidth(openBracketToken); + this.openBracketToken = openBracketToken; + this.AdjustFlagsAndWidth(closeBracketToken); + this.closeBracketToken = closeBracketToken; + this.AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } + + /// SyntaxToken representing the stackalloc keyword. + public SyntaxToken StackAllocKeyword { get { return this.stackAllocKeyword; } } + /// SyntaxToken representing the open bracket. + public SyntaxToken OpenBracketToken { get { return this.openBracketToken; } } + /// SyntaxToken representing the close bracket. + public SyntaxToken CloseBracketToken { get { return this.closeBracketToken; } } + /// InitializerExpressionSyntax representing the initializer expression of the implicit stackalloc array creation expression. + public InitializerExpressionSyntax Initializer { get { return this.initializer; } } + + internal override GreenNode GetSlot(int index) + { + switch (index) + { + case 0: return this.stackAllocKeyword; + case 1: return this.openBracketToken; + case 2: return this.closeBracketToken; + case 3: return this.initializer; + default: return null; + } + } + + internal override SyntaxNode CreateRed(SyntaxNode parent, int position) + { + return new CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax(this, parent, position); + } + + public override TResult Accept(CSharpSyntaxVisitor visitor) + { + return visitor.VisitImplicitStackAllocArrayCreationExpression(this); + } + + public override void Accept(CSharpSyntaxVisitor visitor) + { + visitor.VisitImplicitStackAllocArrayCreationExpression(this); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + { + if (stackAllocKeyword != this.StackAllocKeyword || openBracketToken != this.OpenBracketToken || closeBracketToken != this.CloseBracketToken || initializer != this.Initializer) + { + var newNode = SyntaxFactory.ImplicitStackAllocArrayCreationExpression(stackAllocKeyword, openBracketToken, closeBracketToken, initializer); + var diags = this.GetDiagnostics(); + if (diags != null && diags.Length > 0) + newNode = newNode.WithDiagnosticsGreen(diags); + var annotations = this.GetAnnotations(); + if (annotations != null && annotations.Length > 0) + newNode = newNode.WithAnnotationsGreen(annotations); + return newNode; + } + + return this; + } + + internal override GreenNode SetDiagnostics(DiagnosticInfo[] diagnostics) + { + return new ImplicitStackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.openBracketToken, this.closeBracketToken, this.initializer, diagnostics, GetAnnotations()); + } + + internal override GreenNode SetAnnotations(SyntaxAnnotation[] annotations) + { + return new ImplicitStackAllocArrayCreationExpressionSyntax(this.Kind, this.stackAllocKeyword, this.openBracketToken, this.closeBracketToken, this.initializer, GetDiagnostics(), annotations); + } + + internal ImplicitStackAllocArrayCreationExpressionSyntax(ObjectReader reader) + : base(reader) + { + this.SlotCount = 4; + var stackAllocKeyword = (SyntaxToken)reader.ReadValue(); + if (stackAllocKeyword != null) + { + AdjustFlagsAndWidth(stackAllocKeyword); + this.stackAllocKeyword = stackAllocKeyword; + } + var openBracketToken = (SyntaxToken)reader.ReadValue(); + if (openBracketToken != null) + { + AdjustFlagsAndWidth(openBracketToken); + this.openBracketToken = openBracketToken; + } + var closeBracketToken = (SyntaxToken)reader.ReadValue(); + if (closeBracketToken != null) + { + AdjustFlagsAndWidth(closeBracketToken); + this.closeBracketToken = closeBracketToken; + } + var initializer = (InitializerExpressionSyntax)reader.ReadValue(); + if (initializer != null) + { + AdjustFlagsAndWidth(initializer); + this.initializer = initializer; + } + } + + internal override void WriteTo(ObjectWriter writer) + { + base.WriteTo(writer); + writer.WriteValue(this.stackAllocKeyword); + writer.WriteValue(this.openBracketToken); + writer.WriteValue(this.closeBracketToken); + writer.WriteValue(this.initializer); + } + + static ImplicitStackAllocArrayCreationExpressionSyntax() + { + ObjectBinder.RegisterTypeReader(typeof(ImplicitStackAllocArrayCreationExpressionSyntax), r => new ImplicitStackAllocArrayCreationExpressionSyntax(r)); + } + } + internal abstract partial class QueryClauseSyntax : CSharpSyntaxNode { internal QueryClauseSyntax(SyntaxKind kind, DiagnosticInfo[] diagnostics, SyntaxAnnotation[] annotations) @@ -33893,6 +34080,11 @@ public virtual TResult VisitStackAllocArrayCreationExpression(StackAllocArrayCre return this.DefaultVisit(node); } + public virtual TResult VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + return this.DefaultVisit(node); + } + public virtual TResult VisitQueryExpression(QueryExpressionSyntax node) { return this.DefaultVisit(node); @@ -34917,6 +35109,11 @@ public virtual void VisitStackAllocArrayCreationExpression(StackAllocArrayCreati this.DefaultVisit(node); } + public virtual void VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + this.DefaultVisit(node); + } + public virtual void VisitQueryExpression(QueryExpressionSyntax node) { this.DefaultVisit(node); @@ -36092,7 +36289,17 @@ public override CSharpSyntaxNode VisitStackAllocArrayCreationExpression(StackAll { var stackAllocKeyword = (SyntaxToken)this.Visit(node.StackAllocKeyword); var type = (TypeSyntax)this.Visit(node.Type); - return node.Update(stackAllocKeyword, type); + var initializer = (InitializerExpressionSyntax)this.Visit(node.Initializer); + return node.Update(stackAllocKeyword, type, initializer); + } + + public override CSharpSyntaxNode VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + var stackAllocKeyword = (SyntaxToken)this.Visit(node.StackAllocKeyword); + var openBracketToken = (SyntaxToken)this.Visit(node.OpenBracketToken); + var closeBracketToken = (SyntaxToken)this.Visit(node.CloseBracketToken); + var initializer = (InitializerExpressionSyntax)this.Visit(node.Initializer); + return node.Update(stackAllocKeyword, openBracketToken, closeBracketToken, initializer); } public override CSharpSyntaxNode VisitQueryExpression(QueryExpressionSyntax node) @@ -39282,7 +39489,7 @@ public ImplicitArrayCreationExpressionSyntax ImplicitArrayCreationExpression(Syn return new ImplicitArrayCreationExpressionSyntax(SyntaxKind.ImplicitArrayCreationExpression, newKeyword, openBracketToken, commas.Node, closeBracketToken, initializer, this.context); } - public StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type) + public StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) { #if DEBUG if (stackAllocKeyword == null) @@ -39299,10 +39506,10 @@ public StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression #endif int hash; - var cached = CSharpSyntaxNodeCache.TryGetNode((int)SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, this.context, out hash); + var cached = CSharpSyntaxNodeCache.TryGetNode((int)SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, initializer, this.context, out hash); if (cached != null) return (StackAllocArrayCreationExpressionSyntax)cached; - var result = new StackAllocArrayCreationExpressionSyntax(SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, this.context); + var result = new StackAllocArrayCreationExpressionSyntax(SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, initializer, this.context); if (hash >= 0) { SyntaxNodeCache.AddNode(result, hash); @@ -39311,6 +39518,43 @@ public StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression return result; } + public ImplicitStackAllocArrayCreationExpressionSyntax ImplicitStackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + { +#if DEBUG + if (stackAllocKeyword == null) + throw new ArgumentNullException(nameof(stackAllocKeyword)); + switch (stackAllocKeyword.Kind) + { + case SyntaxKind.StackAllocKeyword: + break; + default: + throw new ArgumentException("stackAllocKeyword"); + } + if (openBracketToken == null) + throw new ArgumentNullException(nameof(openBracketToken)); + switch (openBracketToken.Kind) + { + case SyntaxKind.OpenBracketToken: + break; + default: + throw new ArgumentException("openBracketToken"); + } + if (closeBracketToken == null) + throw new ArgumentNullException(nameof(closeBracketToken)); + switch (closeBracketToken.Kind) + { + case SyntaxKind.CloseBracketToken: + break; + default: + throw new ArgumentException("closeBracketToken"); + } + if (initializer == null) + throw new ArgumentNullException(nameof(initializer)); +#endif + + return new ImplicitStackAllocArrayCreationExpressionSyntax(SyntaxKind.ImplicitStackAllocArrayCreationExpression, stackAllocKeyword, openBracketToken, closeBracketToken, initializer, this.context); + } + public QueryExpressionSyntax QueryExpression(FromClauseSyntax fromClause, QueryBodySyntax body) { #if DEBUG @@ -46208,7 +46452,7 @@ public static ImplicitArrayCreationExpressionSyntax ImplicitArrayCreationExpress return new ImplicitArrayCreationExpressionSyntax(SyntaxKind.ImplicitArrayCreationExpression, newKeyword, openBracketToken, commas.Node, closeBracketToken, initializer); } - public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type) + public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) { #if DEBUG if (stackAllocKeyword == null) @@ -46225,10 +46469,10 @@ public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExp #endif int hash; - var cached = SyntaxNodeCache.TryGetNode((int)SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, out hash); + var cached = SyntaxNodeCache.TryGetNode((int)SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, initializer, out hash); if (cached != null) return (StackAllocArrayCreationExpressionSyntax)cached; - var result = new StackAllocArrayCreationExpressionSyntax(SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type); + var result = new StackAllocArrayCreationExpressionSyntax(SyntaxKind.StackAllocArrayCreationExpression, stackAllocKeyword, type, initializer); if (hash >= 0) { SyntaxNodeCache.AddNode(result, hash); @@ -46237,6 +46481,43 @@ public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExp return result; } + public static ImplicitStackAllocArrayCreationExpressionSyntax ImplicitStackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + { +#if DEBUG + if (stackAllocKeyword == null) + throw new ArgumentNullException(nameof(stackAllocKeyword)); + switch (stackAllocKeyword.Kind) + { + case SyntaxKind.StackAllocKeyword: + break; + default: + throw new ArgumentException("stackAllocKeyword"); + } + if (openBracketToken == null) + throw new ArgumentNullException(nameof(openBracketToken)); + switch (openBracketToken.Kind) + { + case SyntaxKind.OpenBracketToken: + break; + default: + throw new ArgumentException("openBracketToken"); + } + if (closeBracketToken == null) + throw new ArgumentNullException(nameof(closeBracketToken)); + switch (closeBracketToken.Kind) + { + case SyntaxKind.CloseBracketToken: + break; + default: + throw new ArgumentException("closeBracketToken"); + } + if (initializer == null) + throw new ArgumentNullException(nameof(initializer)); +#endif + + return new ImplicitStackAllocArrayCreationExpressionSyntax(SyntaxKind.ImplicitStackAllocArrayCreationExpression, stackAllocKeyword, openBracketToken, closeBracketToken, initializer); + } + public static QueryExpressionSyntax QueryExpression(FromClauseSyntax fromClause, QueryBodySyntax body) { #if DEBUG @@ -51326,6 +51607,7 @@ internal static IEnumerable GetNodeTypes() typeof(ArrayCreationExpressionSyntax), typeof(ImplicitArrayCreationExpressionSyntax), typeof(StackAllocArrayCreationExpressionSyntax), + typeof(ImplicitStackAllocArrayCreationExpressionSyntax), typeof(QueryExpressionSyntax), typeof(QueryBodySyntax), typeof(FromClauseSyntax), diff --git a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Main.Generated.cs b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Main.Generated.cs index 5d57fa01cf02f..926f6c6cc957e 100644 --- a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Main.Generated.cs +++ b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Main.Generated.cs @@ -352,6 +352,12 @@ public virtual TResult VisitStackAllocArrayCreationExpression(StackAllocArrayCre return this.DefaultVisit(node); } + /// Called when the visitor visits a ImplicitStackAllocArrayCreationExpressionSyntax node. + public virtual TResult VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + return this.DefaultVisit(node); + } + /// Called when the visitor visits a QueryExpressionSyntax node. public virtual TResult VisitQueryExpression(QueryExpressionSyntax node) { @@ -1579,6 +1585,12 @@ public virtual void VisitStackAllocArrayCreationExpression(StackAllocArrayCreati this.DefaultVisit(node); } + /// Called when the visitor visits a ImplicitStackAllocArrayCreationExpressionSyntax node. + public virtual void VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + this.DefaultVisit(node); + } + /// Called when the visitor visits a QueryExpressionSyntax node. public virtual void VisitQueryExpression(QueryExpressionSyntax node) { @@ -2902,7 +2914,17 @@ public override SyntaxNode VisitStackAllocArrayCreationExpression(StackAllocArra { var stackAllocKeyword = this.VisitToken(node.StackAllocKeyword); var type = (TypeSyntax)this.Visit(node.Type); - return node.Update(stackAllocKeyword, type); + var initializer = (InitializerExpressionSyntax)this.Visit(node.Initializer); + return node.Update(stackAllocKeyword, type, initializer); + } + + public override SyntaxNode VisitImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node) + { + var stackAllocKeyword = this.VisitToken(node.StackAllocKeyword); + var openBracketToken = this.VisitToken(node.OpenBracketToken); + var closeBracketToken = this.VisitToken(node.CloseBracketToken); + var initializer = (InitializerExpressionSyntax)this.Visit(node.Initializer); + return node.Update(stackAllocKeyword, openBracketToken, closeBracketToken, initializer); } public override SyntaxNode VisitQueryExpression(QueryExpressionSyntax node) @@ -5994,7 +6016,7 @@ public static ImplicitArrayCreationExpressionSyntax ImplicitArrayCreationExpress } /// Creates a new StackAllocArrayCreationExpressionSyntax instance. - public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type) + public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) { switch (stackAllocKeyword.Kind()) { @@ -6005,14 +6027,56 @@ public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExp } if (type == null) throw new ArgumentNullException(nameof(type)); - return (StackAllocArrayCreationExpressionSyntax)Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.StackAllocArrayCreationExpression((Syntax.InternalSyntax.SyntaxToken)stackAllocKeyword.Node, type == null ? null : (Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.TypeSyntax)type.Green).CreateRed(); + return (StackAllocArrayCreationExpressionSyntax)Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.StackAllocArrayCreationExpression((Syntax.InternalSyntax.SyntaxToken)stackAllocKeyword.Node, type == null ? null : (Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.TypeSyntax)type.Green, initializer == null ? null : (Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.InitializerExpressionSyntax)initializer.Green).CreateRed(); } + /// Creates a new StackAllocArrayCreationExpressionSyntax instance. + public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(TypeSyntax type, InitializerExpressionSyntax initializer) + { + return SyntaxFactory.StackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), type, initializer); + } + /// Creates a new StackAllocArrayCreationExpressionSyntax instance. public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(TypeSyntax type) { - return SyntaxFactory.StackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), type); + return SyntaxFactory.StackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), type, default(InitializerExpressionSyntax)); + } + + /// Creates a new ImplicitStackAllocArrayCreationExpressionSyntax instance. + public static ImplicitStackAllocArrayCreationExpressionSyntax ImplicitStackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + { + switch (stackAllocKeyword.Kind()) + { + case SyntaxKind.StackAllocKeyword: + break; + default: + throw new ArgumentException("stackAllocKeyword"); + } + switch (openBracketToken.Kind()) + { + case SyntaxKind.OpenBracketToken: + break; + default: + throw new ArgumentException("openBracketToken"); + } + switch (closeBracketToken.Kind()) + { + case SyntaxKind.CloseBracketToken: + break; + default: + throw new ArgumentException("closeBracketToken"); + } + if (initializer == null) + throw new ArgumentNullException(nameof(initializer)); + return (ImplicitStackAllocArrayCreationExpressionSyntax)Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.ImplicitStackAllocArrayCreationExpression((Syntax.InternalSyntax.SyntaxToken)stackAllocKeyword.Node, (Syntax.InternalSyntax.SyntaxToken)openBracketToken.Node, (Syntax.InternalSyntax.SyntaxToken)closeBracketToken.Node, initializer == null ? null : (Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.InitializerExpressionSyntax)initializer.Green).CreateRed(); + } + + + /// Creates a new ImplicitStackAllocArrayCreationExpressionSyntax instance. + public static ImplicitStackAllocArrayCreationExpressionSyntax ImplicitStackAllocArrayCreationExpression(InitializerExpressionSyntax initializer) + { + return SyntaxFactory.ImplicitStackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), SyntaxFactory.Token(SyntaxKind.OpenBracketToken), SyntaxFactory.Token(SyntaxKind.CloseBracketToken), initializer); } /// Creates a new QueryExpressionSyntax instance. diff --git a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Syntax.Generated.cs b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Syntax.Generated.cs index 9dcf7e032c4f9..106f7a6daa664 100644 --- a/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Syntax.Generated.cs +++ b/src/Compilers/CSharp/Portable/Generated/Syntax.xml.Syntax.Generated.cs @@ -5081,6 +5081,7 @@ public ImplicitArrayCreationExpressionSyntax AddInitializerExpressions(params Ex public sealed partial class StackAllocArrayCreationExpressionSyntax : ExpressionSyntax { private TypeSyntax type; + private InitializerExpressionSyntax initializer; internal StackAllocArrayCreationExpressionSyntax(Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.CSharpSyntaxNode green, SyntaxNode parent, int position) : base(green, parent, position) @@ -5102,11 +5103,21 @@ public TypeSyntax Type } } + /// InitializerExpressionSyntax node representing the initializer of the stackalloc array creation expression. + public InitializerExpressionSyntax Initializer + { + get + { + return this.GetRed(ref this.initializer, 2); + } + } + internal override SyntaxNode GetNodeSlot(int index) { switch (index) { case 1: return this.GetRed(ref this.type, 1); + case 2: return this.GetRed(ref this.initializer, 2); default: return null; } } @@ -5115,6 +5126,7 @@ internal override SyntaxNode GetCachedSlot(int index) switch (index) { case 1: return this.type; + case 2: return this.initializer; default: return null; } } @@ -5129,11 +5141,11 @@ public override void Accept(CSharpSyntaxVisitor visitor) visitor.VisitStackAllocArrayCreationExpression(this); } - public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, TypeSyntax type) + public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, TypeSyntax type, InitializerExpressionSyntax initializer) { - if (stackAllocKeyword != this.StackAllocKeyword || type != this.Type) + if (stackAllocKeyword != this.StackAllocKeyword || type != this.Type || initializer != this.Initializer) { - var newNode = SyntaxFactory.StackAllocArrayCreationExpression(stackAllocKeyword, type); + var newNode = SyntaxFactory.StackAllocArrayCreationExpression(stackAllocKeyword, type, initializer); var annotations = this.GetAnnotations(); if (annotations != null && annotations.Length > 0) return newNode.WithAnnotations(annotations); @@ -5145,12 +5157,121 @@ public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyw public StackAllocArrayCreationExpressionSyntax WithStackAllocKeyword(SyntaxToken stackAllocKeyword) { - return this.Update(stackAllocKeyword, this.Type); + return this.Update(stackAllocKeyword, this.Type, this.Initializer); } public StackAllocArrayCreationExpressionSyntax WithType(TypeSyntax type) { - return this.Update(this.StackAllocKeyword, type); + return this.Update(this.StackAllocKeyword, type, this.Initializer); + } + + public StackAllocArrayCreationExpressionSyntax WithInitializer(InitializerExpressionSyntax initializer) + { + return this.Update(this.StackAllocKeyword, this.Type, initializer); + } + } + + /// Class which represents the syntax node for implicit stackalloc array creation expression. + public sealed partial class ImplicitStackAllocArrayCreationExpressionSyntax : ExpressionSyntax + { + private InitializerExpressionSyntax initializer; + + internal ImplicitStackAllocArrayCreationExpressionSyntax(Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.CSharpSyntaxNode green, SyntaxNode parent, int position) + : base(green, parent, position) + { + } + + /// SyntaxToken representing the stackalloc keyword. + public SyntaxToken StackAllocKeyword + { + get { return new SyntaxToken(this, ((Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.ImplicitStackAllocArrayCreationExpressionSyntax)this.Green).stackAllocKeyword, this.Position, 0); } + } + + /// SyntaxToken representing the open bracket. + public SyntaxToken OpenBracketToken + { + get { return new SyntaxToken(this, ((Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.ImplicitStackAllocArrayCreationExpressionSyntax)this.Green).openBracketToken, this.GetChildPosition(1), this.GetChildIndex(1)); } + } + + /// SyntaxToken representing the close bracket. + public SyntaxToken CloseBracketToken + { + get { return new SyntaxToken(this, ((Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.ImplicitStackAllocArrayCreationExpressionSyntax)this.Green).closeBracketToken, this.GetChildPosition(2), this.GetChildIndex(2)); } + } + + /// InitializerExpressionSyntax representing the initializer expression of the implicit stackalloc array creation expression. + public InitializerExpressionSyntax Initializer + { + get + { + return this.GetRed(ref this.initializer, 3); + } + } + + internal override SyntaxNode GetNodeSlot(int index) + { + switch (index) + { + case 3: return this.GetRed(ref this.initializer, 3); + default: return null; + } + } + internal override SyntaxNode GetCachedSlot(int index) + { + switch (index) + { + case 3: return this.initializer; + default: return null; + } + } + + public override TResult Accept(CSharpSyntaxVisitor visitor) + { + return visitor.VisitImplicitStackAllocArrayCreationExpression(this); + } + + public override void Accept(CSharpSyntaxVisitor visitor) + { + visitor.VisitImplicitStackAllocArrayCreationExpression(this); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, SyntaxToken openBracketToken, SyntaxToken closeBracketToken, InitializerExpressionSyntax initializer) + { + if (stackAllocKeyword != this.StackAllocKeyword || openBracketToken != this.OpenBracketToken || closeBracketToken != this.CloseBracketToken || initializer != this.Initializer) + { + var newNode = SyntaxFactory.ImplicitStackAllocArrayCreationExpression(stackAllocKeyword, openBracketToken, closeBracketToken, initializer); + var annotations = this.GetAnnotations(); + if (annotations != null && annotations.Length > 0) + return newNode.WithAnnotations(annotations); + return newNode; + } + + return this; + } + + public ImplicitStackAllocArrayCreationExpressionSyntax WithStackAllocKeyword(SyntaxToken stackAllocKeyword) + { + return this.Update(stackAllocKeyword, this.OpenBracketToken, this.CloseBracketToken, this.Initializer); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax WithOpenBracketToken(SyntaxToken openBracketToken) + { + return this.Update(this.StackAllocKeyword, openBracketToken, this.CloseBracketToken, this.Initializer); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax WithCloseBracketToken(SyntaxToken closeBracketToken) + { + return this.Update(this.StackAllocKeyword, this.OpenBracketToken, closeBracketToken, this.Initializer); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax WithInitializer(InitializerExpressionSyntax initializer) + { + return this.Update(this.StackAllocKeyword, this.OpenBracketToken, this.CloseBracketToken, initializer); + } + + public ImplicitStackAllocArrayCreationExpressionSyntax AddInitializerExpressions(params ExpressionSyntax[] items) + { + return this.WithInitializer(this.Initializer.WithExpressions(this.Initializer.Expressions.AddRange(items))); } } diff --git a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs index dc51684f6a1d7..2434ab11f1d6a 100644 --- a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs +++ b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs @@ -19,6 +19,11 @@ public override BoundNode VisitPointerElementAccess(BoundPointerElementAccess no } private BoundExpression RewritePointerElementAccess(BoundPointerElementAccess node, BoundExpression rewrittenExpression, BoundExpression rewrittenIndex) + { + return RewritePointerElementAccess(node, rewrittenExpression, rewrittenIndex, node.Checked); + } + + private BoundExpression RewritePointerElementAccess(BoundExpression node, BoundExpression rewrittenExpression, BoundExpression rewrittenIndex, bool @checked) { // Optimization: p[0] == *p if (rewrittenIndex.IsDefaultValue()) @@ -49,7 +54,7 @@ private BoundExpression RewritePointerElementAccess(BoundPointerElementAccess no throw ExceptionUtilities.UnexpectedValue(rewrittenIndex.Type.SpecialType); } - if (node.Checked) + if (@checked) { additionKind |= BinaryOperatorKind.Checked; } diff --git a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_StackAlloc.cs b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_StackAlloc.cs index a4f295d554120..61e5e87dd5b32 100644 --- a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_StackAlloc.cs +++ b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_StackAlloc.cs @@ -27,17 +27,23 @@ public override BoundNode VisitStackAllocArrayCreation(BoundStackAllocArrayCreat var elementType = stackAllocNode.ElementType; + var initializerOpt = stackAllocNode.InitializerOpt; + if (initializerOpt != null) + { + initializerOpt = initializerOpt.Update(VisitList(initializerOpt.Initializers)); + } + if (type.IsPointerType()) { var stackSize = RewriteStackAllocCountToSize(rewrittenCount, elementType); - return new BoundConvertedStackAllocExpression(stackAllocNode.Syntax, elementType, stackSize, stackAllocNode.Type); + return new BoundConvertedStackAllocExpression(stackAllocNode.Syntax, elementType, stackSize, initializerOpt, stackAllocNode.Type); } else if (type.OriginalDefinition == _compilation.GetWellKnownType(WellKnownType.System_Span_T)) { var spanType = (NamedTypeSymbol)stackAllocNode.Type; var countTemp = _factory.StoreToTemp(rewrittenCount, out BoundAssignmentOperator countTempAssignment); var stackSize = RewriteStackAllocCountToSize(countTemp, elementType); - stackAllocNode = new BoundConvertedStackAllocExpression(stackAllocNode.Syntax, elementType, stackSize, spanType); + stackAllocNode = new BoundConvertedStackAllocExpression(stackAllocNode.Syntax, elementType, stackSize, initializerOpt, spanType); var spanCtor = (MethodSymbol)_compilation.GetWellKnownTypeMember(WellKnownMember.System_Span_T__ctor).SymbolAsMember(spanType); var ctorCall = _factory.New(spanCtor, stackAllocNode, countTemp); diff --git a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs index 7a7803df024bf..4b73468bfb623 100644 --- a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs +++ b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs @@ -10309,7 +10309,7 @@ private ExpressionSyntax ParseArrayOrObjectCreationExpression() { initializer = this.ParseArrayInitializer(); } - else if (type.Kind == SyntaxKind.ArrayType) + else { var rankSpec = ((ArrayTypeSyntax)type).RankSpecifiers[0]; if (GetNumberOfNonOmittedArraySizes(rankSpec) == 0) @@ -10585,10 +10585,11 @@ private void ParseExpressionsForComplexElementInitializer(ref SyntaxToken openBr } } } - + private bool IsImplicitlyTypedArray() { - return this.CurrentToken.Kind == SyntaxKind.NewKeyword && this.PeekToken(1).Kind == SyntaxKind.OpenBracketToken; + Debug.Assert(this.CurrentToken.Kind == SyntaxKind.NewKeyword || this.CurrentToken.Kind == SyntaxKind.StackAllocKeyword); + return this.PeekToken(1).Kind == SyntaxKind.OpenBracketToken; } private ImplicitArrayCreationExpressionSyntax ParseImplicitlyTypedArrayCreation() @@ -10685,11 +10686,50 @@ private PostSkipAction SkipBadArrayInitializerTokens(ref SyntaxToken openBrace, expected); } - private StackAllocArrayCreationExpressionSyntax ParseStackAllocExpression() + private ExpressionSyntax ParseStackAllocExpression() { - var stackAlloc = this.EatToken(SyntaxKind.StackAllocKeyword); + if (this.IsImplicitlyTypedArray()) + { + return ParseImplicitlyTypedStackAllocExpression(); + } + else + { + return ParseRegularStackAllocExpression(); + } + } + + private ExpressionSyntax ParseImplicitlyTypedStackAllocExpression() + { + var @stackalloc = this.EatToken(SyntaxKind.StackAllocKeyword); + @stackalloc = CheckFeatureAvailability(@stackalloc, MessageID.IDS_FeatureStackAllocInitializer); + var openBracket = this.EatToken(SyntaxKind.OpenBracketToken); + var closeBracket = this.EatToken(SyntaxKind.CloseBracketToken); + var initializer = this.ParseArrayInitializer(); + return _syntaxFactory.ImplicitStackAllocArrayCreationExpression(@stackalloc, openBracket, closeBracket, initializer); + } + + private ExpressionSyntax ParseRegularStackAllocExpression() + { + var @stackalloc = this.EatToken(SyntaxKind.StackAllocKeyword); var elementType = this.ParseType(expectSizes: true); - return _syntaxFactory.StackAllocArrayCreationExpression(stackAlloc, elementType); + InitializerExpressionSyntax initializer; + if (this.CurrentToken.Kind == SyntaxKind.OpenBraceToken) + { + @stackalloc = CheckFeatureAvailability(@stackalloc, MessageID.IDS_FeatureStackAllocInitializer); + initializer = this.ParseArrayInitializer(); + } + else + { + var rankSpec = ((ArrayTypeSyntax)elementType).RankSpecifiers[0]; + if (GetNumberOfNonOmittedArraySizes(rankSpec) == 0) + { + elementType = this.AddError(elementType, rankSpec, ErrorCode.ERR_MissingArraySize); + } + + initializer = null; + } + + return _syntaxFactory.StackAllocArrayCreationExpression(@stackalloc, elementType, initializer); } private AnonymousMethodExpressionSyntax ParseAnonymousMethodExpression() diff --git a/src/Compilers/CSharp/Portable/PublicAPI.Unshipped.txt b/src/Compilers/CSharp/Portable/PublicAPI.Unshipped.txt index ac23b0502c300..4a0d7bfbb383a 100644 --- a/src/Compilers/CSharp/Portable/PublicAPI.Unshipped.txt +++ b/src/Compilers/CSharp/Portable/PublicAPI.Unshipped.txt @@ -17,9 +17,27 @@ Microsoft.CodeAnalysis.CSharp.Syntax.ArgumentSyntax.WithRefKindKeyword(Microsoft Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax.RefKindKeyword.get -> Microsoft.CodeAnalysis.SyntaxToken Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax.Update(Microsoft.CodeAnalysis.SyntaxToken refKindKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type) -> Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax.WithRefKindKeyword(Microsoft.CodeAnalysis.SyntaxToken refKindKeyword) -> Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.AddInitializerExpressions(params Microsoft.CodeAnalysis.CSharp.Syntax.ExpressionSyntax[] items) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.CloseBracketToken.get -> Microsoft.CodeAnalysis.SyntaxToken +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.Initializer.get -> Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.OpenBracketToken.get -> Microsoft.CodeAnalysis.SyntaxToken +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.StackAllocKeyword.get -> Microsoft.CodeAnalysis.SyntaxToken +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.Update(Microsoft.CodeAnalysis.SyntaxToken stackAllocKeyword, Microsoft.CodeAnalysis.SyntaxToken openBracketToken, Microsoft.CodeAnalysis.SyntaxToken closeBracketToken, Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.WithCloseBracketToken(Microsoft.CodeAnalysis.SyntaxToken closeBracketToken) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.WithInitializer(Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.WithOpenBracketToken(Microsoft.CodeAnalysis.SyntaxToken openBracketToken) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.WithStackAllocKeyword(Microsoft.CodeAnalysis.SyntaxToken stackAllocKeyword) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax Microsoft.CodeAnalysis.CSharp.Syntax.RefTypeSyntax.ReadOnlyKeyword.get -> Microsoft.CodeAnalysis.SyntaxToken Microsoft.CodeAnalysis.CSharp.Syntax.RefTypeSyntax.Update(Microsoft.CodeAnalysis.SyntaxToken refKeyword, Microsoft.CodeAnalysis.SyntaxToken readOnlyKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type) -> Microsoft.CodeAnalysis.CSharp.Syntax.RefTypeSyntax Microsoft.CodeAnalysis.CSharp.Syntax.RefTypeSyntax.WithReadOnlyKeyword(Microsoft.CodeAnalysis.SyntaxToken readOnlyKeyword) -> Microsoft.CodeAnalysis.CSharp.Syntax.RefTypeSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax.Initializer.get -> Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax.Update(Microsoft.CodeAnalysis.SyntaxToken stackAllocKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type, Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax.WithInitializer(Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax +Microsoft.CodeAnalysis.CSharp.SyntaxKind.ImplicitStackAllocArrayCreationExpression = 9053 -> Microsoft.CodeAnalysis.CSharp.SyntaxKind +override Microsoft.CodeAnalysis.CSharp.CSharpSyntaxRewriter.VisitImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax node) -> Microsoft.CodeAnalysis.SyntaxNode +override Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.Accept(Microsoft.CodeAnalysis.CSharp.CSharpSyntaxVisitor visitor) -> void +override Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax.Accept(Microsoft.CodeAnalysis.CSharp.CSharpSyntaxVisitor visitor) -> TResult static Microsoft.CodeAnalysis.CSharp.CSharpCommandLineParser.Script.get -> Microsoft.CodeAnalysis.CSharp.CSharpCommandLineParser static Microsoft.CodeAnalysis.CSharp.CSharpExtensions.GetConversion(this Microsoft.CodeAnalysis.Operations.IConversionOperation conversionExpression) -> Microsoft.CodeAnalysis.CSharp.Conversion static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.Argument(Microsoft.CodeAnalysis.CSharp.Syntax.NameColonSyntax nameColon, Microsoft.CodeAnalysis.SyntaxToken refKindKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.ExpressionSyntax expression) -> Microsoft.CodeAnalysis.CSharp.Syntax.ArgumentSyntax @@ -28,4 +46,10 @@ static Microsoft.CodeAnalysis.CSharp.CSharpExtensions.GetDeconstructionInfo(this static Microsoft.CodeAnalysis.CSharp.CSharpExtensions.GetDeconstructionInfo(this Microsoft.CodeAnalysis.SemanticModel semanticModel, Microsoft.CodeAnalysis.CSharp.Syntax.ForEachVariableStatementSyntax foreach) -> Microsoft.CodeAnalysis.CSharp.DeconstructionInfo static Microsoft.CodeAnalysis.CSharp.CSharpExtensions.GetInConversion(this Microsoft.CodeAnalysis.Operations.ICompoundAssignmentOperation compoundAssignment) -> Microsoft.CodeAnalysis.CSharp.Conversion static Microsoft.CodeAnalysis.CSharp.CSharpExtensions.GetOutConversion(this Microsoft.CodeAnalysis.Operations.ICompoundAssignmentOperation compoundAssignment) -> Microsoft.CodeAnalysis.CSharp.Conversion -static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.CrefParameter(Microsoft.CodeAnalysis.SyntaxToken refKindKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type) -> Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax \ No newline at end of file +static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.CrefParameter(Microsoft.CodeAnalysis.SyntaxToken refKindKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type) -> Microsoft.CodeAnalysis.CSharp.Syntax.CrefParameterSyntax +static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.ImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.ImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.SyntaxToken stackAllocKeyword, Microsoft.CodeAnalysis.SyntaxToken openBracketToken, Microsoft.CodeAnalysis.SyntaxToken closeBracketToken, Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax +static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.StackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type, Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax +static Microsoft.CodeAnalysis.CSharp.SyntaxFactory.StackAllocArrayCreationExpression(Microsoft.CodeAnalysis.SyntaxToken stackAllocKeyword, Microsoft.CodeAnalysis.CSharp.Syntax.TypeSyntax type, Microsoft.CodeAnalysis.CSharp.Syntax.InitializerExpressionSyntax initializer) -> Microsoft.CodeAnalysis.CSharp.Syntax.StackAllocArrayCreationExpressionSyntax +virtual Microsoft.CodeAnalysis.CSharp.CSharpSyntaxVisitor.VisitImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax node) -> void +virtual Microsoft.CodeAnalysis.CSharp.CSharpSyntaxVisitor.VisitImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.ImplicitStackAllocArrayCreationExpressionSyntax node) -> TResult diff --git a/src/Compilers/CSharp/Portable/Syntax/StackAllocArrayCreationExpressionSyntax.cs b/src/Compilers/CSharp/Portable/Syntax/StackAllocArrayCreationExpressionSyntax.cs new file mode 100644 index 0000000000000..3882f81c59ba4 --- /dev/null +++ b/src/Compilers/CSharp/Portable/Syntax/StackAllocArrayCreationExpressionSyntax.cs @@ -0,0 +1,25 @@ +// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using Microsoft.CodeAnalysis.CSharp.Syntax; + +namespace Microsoft.CodeAnalysis.CSharp.Syntax +{ + public partial class StackAllocArrayCreationExpressionSyntax + { + public StackAllocArrayCreationExpressionSyntax Update(SyntaxToken stackAllocKeyword, TypeSyntax type) + { + return Update(StackAllocKeyword, type, default(InitializerExpressionSyntax)); + } + } +} + +namespace Microsoft.CodeAnalysis.CSharp +{ + public partial class SyntaxFactory + { + public static StackAllocArrayCreationExpressionSyntax StackAllocArrayCreationExpression(SyntaxToken stackAllocKeyword, TypeSyntax type) + { + return StackAllocArrayCreationExpression(stackAllocKeyword, type, default(InitializerExpressionSyntax)); + } + } +} diff --git a/src/Compilers/CSharp/Portable/Syntax/Syntax.xml b/src/Compilers/CSharp/Portable/Syntax/Syntax.xml index b74cad86c874d..b6fcc66190ca2 100644 --- a/src/Compilers/CSharp/Portable/Syntax/Syntax.xml +++ b/src/Compilers/CSharp/Portable/Syntax/Syntax.xml @@ -1498,6 +1498,11 @@ TypeSyntax node representing the type of the stackalloc array. + + + InitializerExpressionSyntax node representing the initializer of the stackalloc array creation expression. + + Class which represents the syntax node for stackalloc array creation expression. @@ -1505,6 +1510,38 @@ Creates a StackAllocArrayCreationExpressionSyntax node. + + + + + + SyntaxToken representing the stackalloc keyword. + + + + + + SyntaxToken representing the open bracket. + + + + + + SyntaxToken representing the close bracket. + + + + + InitializerExpressionSyntax representing the initializer expression of the implicit stackalloc array creation expression. + + + + Class which represents the syntax node for implicit stackalloc array creation expression. + + + Creates an ImplicitStackAllocArrayCreationExpressionSyntax node. + + diff --git a/src/Compilers/CSharp/Portable/Syntax/SyntaxKind.cs b/src/Compilers/CSharp/Portable/Syntax/SyntaxKind.cs index 34f006a9866a4..be5f0b8ad005f 100644 --- a/src/Compilers/CSharp/Portable/Syntax/SyntaxKind.cs +++ b/src/Compilers/CSharp/Portable/Syntax/SyntaxKind.cs @@ -563,5 +563,7 @@ public enum SyntaxKind : ushort RefExpression = 9050, RefType = 9051, ThrowExpression = 9052, + + ImplicitStackAllocArrayCreationExpression = 9053, } } diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs new file mode 100644 index 0000000000000..49f51f221f53e --- /dev/null +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -0,0 +1,388 @@ +// Copyright (c) Microsoft Open Technologies, Inc. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using Microsoft.CodeAnalysis.CSharp.Test.Utilities; +using Microsoft.CodeAnalysis.Test.Utilities; +using Xunit; + +namespace Microsoft.CodeAnalysis.CSharp.UnitTests +{ + [CompilerTrait(CompilerFeature.StackAllocInitializer)] + public class StackAllocInitializerTests : CompilingTestBase + { + [Fact] + public void TestUnused() + { + var text = @" +using System; + +static unsafe class C +{ + static byte Method() + { + return 42; + } + + static void Main() + { + byte* p = stackalloc byte[3] { 42, Method(), 42 }; + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Passes).VerifyIL("C.Main", +@"{ + // Code size 10 (0xa) + .maxstack 1 + IL_0000: ldc.i4.3 + IL_0001: conv.u + IL_0002: pop + IL_0003: call ""byte C.Method()"" + IL_0008: pop + IL_0009: ret +}"); + } + + [Fact] + public void TestIdenticalBytes() + { + var text = @" +using System; + +static unsafe class C +{ + static void Print(byte* p) + { + for (int i = 0; i < 3; i++) + Console.Write(p[i]); + } + + static void Main() + { + byte* p = stackalloc byte[3] { 42, 42, 42 }; + Print(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails, expectedOutput: @"424242").VerifyIL("C.Main", +@"{ + // Code size 16 (0x10) + .maxstack 4 + IL_0000: ldc.i4.3 + IL_0001: conv.u + IL_0002: localloc + IL_0004: dup + IL_0005: ldc.i4.s 42 + IL_0007: ldc.i4.3 + IL_0008: initblk + IL_000a: call ""void C.Print(byte*)"" + IL_000f: ret +}"); + } + + [Fact] + public void TestMixedBlockInit() + { + var text = @" +using System; + +static unsafe class C +{ + static void Print(byte* p) + { + for (int i = 0; i < 9; i++) + Console.Write(p[i]); + } + + static byte M() + { + return 9; + } + + static void Main() + { + byte* p = stackalloc byte[9] { 1, 2, 3, 4, 5, 6, 7, 8, M() }; + Print(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails, expectedOutput: @"123456789").VerifyIL("C.Main", +@"{ + // Code size 30 (0x1e) + .maxstack 4 + IL_0000: ldc.i4.s 9 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: ldsflda "".__StaticArrayInitTypeSize=9 .50BE2890A92A315C4BE0FA98C600C8939A260312"" + IL_000b: ldc.i4.s 9 + IL_000d: cpblk + IL_000f: dup + IL_0010: ldc.i4.8 + IL_0011: add + IL_0012: call ""byte C.M()"" + IL_0017: stind.i1 + IL_0018: call ""void C.Print(byte*)"" + IL_001d: ret +}"); + } + + [Fact] + public void TestElementInit() + { + var text = @" +using System; + +struct S +{ + public int i; +} + +static unsafe class C +{ + static void Print(S* p) + { + for (int i = 0; i < 3; i++) + Console.Write(p[i].i); + } + + static S M(int i) + { + return new S { i = i }; + } + + static void Main() + { + S* p = stackalloc S[3] { M(1), M(2), M(3) }; + Print(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails ,expectedOutput: @"123").VerifyIL("C.Main", +@"{ + // Code size 70 (0x46) + .maxstack 4 + IL_0000: ldc.i4.3 + IL_0001: conv.u + IL_0002: sizeof ""S"" + IL_0008: mul.ovf.un + IL_0009: localloc + IL_000b: dup + IL_000c: ldc.i4.1 + IL_000d: call ""S C.M(int)"" + IL_0012: stobj ""S"" + IL_0017: dup + IL_0018: sizeof ""S"" + IL_001e: add + IL_001f: ldc.i4.2 + IL_0020: call ""S C.M(int)"" + IL_0025: stobj ""S"" + IL_002a: dup + IL_002b: ldc.i4.2 + IL_002c: conv.i + IL_002d: sizeof ""S"" + IL_0033: mul + IL_0034: add + IL_0035: ldc.i4.3 + IL_0036: call ""S C.M(int)"" + IL_003b: stobj ""S"" + IL_0040: call ""void C.Print(S*)"" + IL_0045: ret +}"); + } + + [Fact] + public void TestBytePointer() + { + Test("System.Byte", +@"{ + // Code size 19 (0x13) + .maxstack 4 + IL_0000: ldc.i4.3 + IL_0001: conv.u + IL_0002: localloc + IL_0004: dup + IL_0005: ldsflda "".__StaticArrayInitTypeSize=3 .7037807198C22A7D2B0807371D763779A84FDFCF"" + IL_000a: ldc.i4.3 + IL_000b: cpblk + IL_000d: call ""void C.Print(byte*)"" + IL_0012: ret +}"); + } + + [Fact] + public void TestInt32Pointer() + { + Test("System.Int32", +@"{ + // Code size 21 (0x15) + .maxstack 4 + IL_0000: ldc.i4.s 12 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: ldsflda "".__StaticArrayInitTypeSize=12 .E429CCA3F703A39CC5954A6572FEC9086135B34E"" + IL_000b: ldc.i4.s 12 + IL_000d: cpblk + IL_000f: call ""void C.Print(int*)"" + IL_0014: ret +}"); + } + + [Fact] + public void TestInt64Pointer() + { + Test("System.Int64", +@"{ + // Code size 21 (0x15) + .maxstack 4 + IL_0000: ldc.i4.s 24 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: ldsflda "".__StaticArrayInitTypeSize=24 .E2D1839ED1706F7D470D87F8C48A5584CAFA5A12"" + IL_000b: ldc.i4.s 24 + IL_000d: cpblk + IL_000f: call ""void C.Print(long*)"" + IL_0014: ret +}"); + } + + [Fact] + public void TestSpan() + { + var comp = CreateStandardCompilation(@" +using System; +static class C +{ + static void Main() + { + Span p = stackalloc byte[3] { 1, 2, 3 }; + } +} + +namespace System +{ + public ref struct Span { + public unsafe Span(void* p, int length) + { + for (int i = 0; i < 3; i++) + Console.Write(((byte*)p)[i]); + } + } +} +", options: TestOptions.UnsafeReleaseExe, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3)); + CompileAndVerify(comp, verify: Verification.Fails, expectedOutput: @"123") + .VerifyIL("C.Main", +@"{ + // Code size 23 (0x17) + .maxstack 4 + .locals init (int V_0) + IL_0000: ldc.i4.3 + IL_0001: stloc.0 + IL_0002: ldloc.0 + IL_0003: conv.u + IL_0004: localloc + IL_0006: dup + IL_0007: ldsflda "".__StaticArrayInitTypeSize=3 .7037807198C22A7D2B0807371D763779A84FDFCF"" + IL_000c: ldc.i4.3 + IL_000d: cpblk + IL_000f: ldloc.0 + IL_0010: newobj ""System.Span..ctor(void*, int)"" + IL_0015: pop + IL_0016: ret +}"); + } + + [Fact] + public void TestReadOnlySpan() + { + var comp = CreateStandardCompilation(@" +using System; +static class C +{ + static void Main() + { + ReadOnlySpan p = stackalloc int[3] { 1, 2, 3 }; + } +} + +namespace System +{ + public ref struct Span + { + public unsafe Span(void* p, int length) + { + for (int i = 0; i < 3; i++) + Console.Write(((int*)p)[i]); + } + } + + public readonly ref struct ReadOnlySpan + { + public static implicit operator System.ReadOnlySpan (System.Span span) => default; + } +} +", options: TestOptions.UnsafeReleaseExe, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3)); + CompileAndVerify(comp, verify: Verification.Fails, expectedOutput: @"123") + .VerifyIL("C.Main", +@"{ + // Code size 31 (0x1f) + .maxstack 4 + .locals init (int V_0) + IL_0000: ldc.i4.3 + IL_0001: stloc.0 + IL_0002: ldloc.0 + IL_0003: conv.u + IL_0004: ldc.i4.4 + IL_0005: mul.ovf.un + IL_0006: localloc + IL_0008: dup + IL_0009: ldsflda "".__StaticArrayInitTypeSize=12 .E429CCA3F703A39CC5954A6572FEC9086135B34E"" + IL_000e: ldc.i4.s 12 + IL_0010: cpblk + IL_0012: ldloc.0 + IL_0013: newobj ""System.Span..ctor(void*, int)"" + IL_0018: call ""System.ReadOnlySpan System.ReadOnlySpan.op_Implicit(System.Span)"" + IL_001d: pop + IL_001e: ret +}"); + } + + private static string GetSource(string pointerType) => $@" +using System; +using T = {pointerType}; +static unsafe class C +{{ + static void Print(T* p) + {{ + for (int i = 0; i < 3; i++) + Console.Write(p[i]); + }} + + static void Main() + {{ + T* p = stackalloc T[3] {{ 1, 2, 3 }}; + Print(p); + }} +}} +"; + + private void Test(string pointerType, string il) + => CompileAndVerify(GetSource(pointerType), + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails, + expectedOutput: @"123").VerifyIL("C.Main", il); + } +} diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs new file mode 100644 index 0000000000000..7a88323d81a9e --- /dev/null +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -0,0 +1,1137 @@ +// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Threading; +using Microsoft.CodeAnalysis.CSharp.Symbols; +using Microsoft.CodeAnalysis.CSharp.Syntax; +using Microsoft.CodeAnalysis.CSharp.Test.Utilities; +using Microsoft.CodeAnalysis.Test.Utilities; +using Roslyn.Test.Utilities; +using Xunit; + +namespace Microsoft.CodeAnalysis.CSharp.UnitTests +{ + [CompilerTrait(CompilerFeature.StackAllocInitializer)] + public class StackAllocInitializerTests : CompilingTestBase + { + [Fact] + public void NoBestType() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + struct A {} + struct B {} + + public void Method1() + { + var obj1 = stackalloc[] { new A(), new B() }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (9,20): error CS0826: No best type found for implicitly-typed array + // var obj1 = stackalloc[] { new A(), new B() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new A(), new B() }").WithLocation(9, 20) + ); + } + + [Fact] + public void BestTypeNumeric() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc[] { 1, 1.2 }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); + Assert.Equal(1, variables.Count()); + + var obj1 = variables.ElementAt(0); + Assert.Equal("obj1", obj1.Identifier.Text); + + var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); + Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); + } + + [Fact] + public void BadBestType() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc[] { """" }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('string') + // var obj1 = stackalloc[] { "" }; + Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(6, 20) + ); + } + + [Fact] + public void WrongLength() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc int[10] { }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,20): error CS0847: An array initializer of length '10' is expected + // var obj1 = stackalloc int[10] { }; + Diagnostic(ErrorCode.ERR_ArrayInitializerIncorrectLength, "stackalloc int[10] { }").WithArguments("10").WithLocation(6, 20) + ); + } + + [Fact] + public void NoInit() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc int[]; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,34): error CS1586: Array creation must have array size or array initializer + // var obj1 = stackalloc int[]; + Diagnostic(ErrorCode.ERR_MissingArraySize, "[]").WithLocation(6, 34) + ); + } + + [Fact] + public void NoBestType2() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc[]{}; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,20): error CS0826: No best type found for implicitly-typed array + // var obj1 = stackalloc[]{}; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[]{}").WithLocation(6, 20) + ); + } + + [Fact] + public void NestedInit() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc[] {{1}}; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,34): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj1 = stackalloc[] {{1}}; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34), + // (6,34): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj1 = stackalloc[] {{1}}; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34) + ); + } + + [Fact] + public void BadRank() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc int[][] { 1 }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,31): error CS1575: A stackalloc expression requires [] after type + // var obj1 = stackalloc int[][] { 1 }; + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[][]").WithLocation(6, 31) + ); + } + + [Fact] + public void BadDimension() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc int[,] { 1 }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,31): error CS1575: A stackalloc expression requires [] after type + // var obj1 = stackalloc int[,] { 1 }; + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[,]").WithLocation(6, 31) + ); + } + + [Fact] + public void ConversionFromPointerStackAlloc_UserDefined_Implicit() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + Test obj1 = stackalloc int[3] { 1, 2, 3 }; + var obj2 = stackalloc int[3] { 1, 2, 3 }; + Span obj3 = stackalloc int[3] { 1, 2, 3 }; + int* obj4 = stackalloc int[3] { 1, 2, 3 }; + double* obj5 = stackalloc int[3] { 1, 2, 3 }; + } + + public void Method2() + { + Test obj1 = stackalloc int[] { 1, 2, 3 }; + var obj2 = stackalloc int[] { 1, 2, 3 }; + Span obj3 = stackalloc int[] { 1, 2, 3 }; + int* obj4 = stackalloc int[] { 1, 2, 3 }; + double* obj5 = stackalloc int[] { 1, 2, 3 }; + } + + public void Method3() + { + Test obj1 = stackalloc[] { 1, 2, 3 }; + var obj2 = stackalloc[] { 1, 2, 3 }; + Span obj3 = stackalloc[] { 1, 2, 3 }; + int* obj4 = stackalloc[] { 1, 2, 3 }; + double* obj5 = stackalloc[] { 1, 2, 3 }; + } + + public static implicit operator Test(int* value) + { + return default(Test); + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + + // (11,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc int[3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(11, 24), + // (20,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc int[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(20, 24), + // (29,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc[] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(29, 24) + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); + Assert.Equal(15, variables.Count()); + + for (int i = 0; i < 15; i += 5) + { + var obj1 = variables.ElementAt(i); + Assert.Equal("obj1", obj1.Identifier.Text); + + var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); + Assert.Equal("Test", obj1Value.ConvertedType.Name); + Assert.Equal(ConversionKind.ImplicitUserDefined, obj1Value.ImplicitConversion.Kind); + + var obj2 = variables.ElementAt(i + 1); + Assert.Equal("obj2", obj2.Identifier.Text); + + var obj2Value = model.GetSemanticInfoSummary(obj2.Initializer.Value); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj2Value.ImplicitConversion.Kind); + + var obj3 = variables.ElementAt(i + 2); + Assert.Equal("obj3", obj3.Identifier.Text); + + var obj3Value = model.GetSemanticInfoSummary(obj3.Initializer.Value); + Assert.Equal("Span", obj3Value.Type.Name); + Assert.Equal("Span", obj3Value.ConvertedType.Name); + Assert.Equal(ConversionKind.Identity, obj3Value.ImplicitConversion.Kind); + + var obj4 = variables.ElementAt(i + 3); + Assert.Equal("obj4", obj4.Identifier.Text); + + var obj4Value = model.GetSemanticInfoSummary(obj4.Initializer.Value); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj4Value.ImplicitConversion.Kind); + + var obj5 = variables.ElementAt(i + 4); + Assert.Equal("obj5", obj5.Identifier.Text); + + var obj5Value = model.GetSemanticInfoSummary(obj5.Initializer.Value); + Assert.Null(obj5Value.Type); + Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj5Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.NoConversion, obj5Value.ImplicitConversion.Kind); + } + } + + [Fact] + public void ConversionFromPointerStackAlloc_UserDefined_Explicit() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + Test obj1 = (Test)stackalloc int[3] { 1, 2, 3 }; + var obj2 = stackalloc int[3] { 1, 2, 3 }; + Span obj3 = stackalloc int[3] { 1, 2, 3 }; + int* obj4 = stackalloc int[3] { 1, 2, 3 }; + double* obj5 = stackalloc int[3] { 1, 2, 3 }; + } + + public void Method2() + { + Test obj1 = (Test)stackalloc int[] { 1, 2, 3 }; + var obj2 = stackalloc int[] { 1, 2, 3 }; + Span obj3 = stackalloc int[] { 1, 2, 3 }; + int* obj4 = stackalloc int[] { 1, 2, 3 }; + double* obj5 = stackalloc int[] { 1, 2, 3 }; + } + + public void Method3() + { + Test obj1 = (Test)stackalloc [] { 1, 2, 3 }; + var obj2 = stackalloc[] { 1, 2, 3 }; + Span obj3 = stackalloc [] { 1, 2, 3 }; + int* obj4 = stackalloc[] { 1, 2, 3 }; + double* obj5 = stackalloc[] { 1, 2, 3 }; + } + + public static explicit operator Test(Span value) + { + return default(Test); + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (11,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc int[3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(11, 24), + // (20,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc int[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(20, 24), + // (29,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. + // double* obj5 = stackalloc[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc[] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(29, 24) + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); + Assert.Equal(15, variables.Count()); + + for (int i = 0; i < 15; i += 5) + { + var obj1 = variables.ElementAt(i); + Assert.Equal("obj1", obj1.Identifier.Text); + Assert.Equal(SyntaxKind.CastExpression, obj1.Initializer.Value.Kind()); + + var obj1Value = model.GetSemanticInfoSummary(((CastExpressionSyntax)obj1.Initializer.Value).Expression); + Assert.Equal("Span", obj1Value.Type.Name); + Assert.Equal("Span", obj1Value.ConvertedType.Name); + Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); + + var obj2 = variables.ElementAt(i + 1); + Assert.Equal("obj2", obj2.Identifier.Text); + + var obj2Value = model.GetSemanticInfoSummary(obj2.Initializer.Value); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj2Value.ImplicitConversion.Kind); + + var obj3 = variables.ElementAt(i + 2); + Assert.Equal("obj3", obj3.Identifier.Text); + + var obj3Value = model.GetSemanticInfoSummary(obj3.Initializer.Value); + Assert.Equal("Span", obj3Value.Type.Name); + Assert.Equal("Span", obj3Value.ConvertedType.Name); + Assert.Equal(ConversionKind.Identity, obj3Value.ImplicitConversion.Kind); + + var obj4 = variables.ElementAt(i + 3); + Assert.Equal("obj4", obj4.Identifier.Text); + + var obj4Value = model.GetSemanticInfoSummary(obj4.Initializer.Value); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj4Value.ImplicitConversion.Kind); + + var obj5 = variables.ElementAt(i + 4); + Assert.Equal("obj5", obj5.Identifier.Text); + + var obj5Value = model.GetSemanticInfoSummary(obj5.Initializer.Value); + Assert.Null(obj5Value.Type); + Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj5Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.NoConversion, obj5Value.ImplicitConversion.Kind); + } + } + + [Fact] + public void ConversionError() + { + CreateCompilationWithMscorlibAndSpan(@" +class Test +{ + void Method1() + { + double x = stackalloc int[3] { 1, 2, 3 }; // implicit + short y = (short)stackalloc int[3] { 1, 2, 3 }; // explicit + } + + void Method2() + { + double x = stackalloc int[] { 1, 2, 3 }; // implicit + short y = (short)stackalloc int[] { 1, 2, 3 }; // explicit + } + + void Method3() + { + double x = stackalloc[] { 1, 2, 3 }; // implicit + short y = (short)stackalloc[] { 1, 2, 3 }; // explicit + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + + + // (6,20): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double' is not possible. + // double x = stackalloc int[3] { 1, 2, 3 }; // implicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "double").WithLocation(6, 20), + // (7,19): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'short' is not possible. + // short y = (short)stackalloc int[3] { 1, 2, 3 }; // explicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(short)stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "short").WithLocation(7, 19), + // (12,20): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double' is not possible. + // double x = stackalloc int[] { 1, 2, 3 }; // implicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[] { 1, 2, 3 }").WithArguments("int", "double").WithLocation(12, 20), + // (13,19): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'short' is not possible. + // short y = (short)stackalloc int[] { 1, 2, 3 }; // explicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(short)stackalloc int[] { 1, 2, 3 }").WithArguments("int", "short").WithLocation(13, 19), + // (18,20): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double' is not possible. + // double x = stackalloc[] { 1, 2, 3 }; // implicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc[] { 1, 2, 3 }").WithArguments("int", "double").WithLocation(18, 20), + // (19,19): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'short' is not possible. + // short y = (short)stackalloc[] { 1, 2, 3 }; // explicit + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(short)stackalloc[] { 1, 2, 3 }").WithArguments("int", "short").WithLocation(19, 19) + ); + } + + [Fact] + public void MissingSpanType() + { + CreateStandardCompilation(@" +class Test +{ + void M() + { + Span a1 = stackalloc int[3] { 1, 2, 3 }; + Span a2 = stackalloc int[] { 1, 2, 3 }; + Span a3 = stackalloc[] { 1, 2, 3 }; + } +}").VerifyDiagnostics( + // (6,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) + // Span a1 = stackalloc int[3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(6, 9), + // (6,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported + // Span a1 = stackalloc int[3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int[3] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(6, 24), + // (7,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) + // Span a2 = stackalloc int[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(7, 9), + // (7,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported + // Span a2 = stackalloc int[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int[] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(7, 24), + // (8,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) + // Span a3 = stackalloc[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(8, 9), + // (8,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported + // Span a3 = stackalloc[] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc[] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(8, 24) + ); + } + + [Fact] + public void MissingSpanConstructor() + { + CreateStandardCompilation(@" +namespace System +{ + ref struct Span + { + } + class Test + { + void M() + { + Span a1 = stackalloc int [3] { 1, 2, 3 }; + Span a2 = stackalloc int [] { 1, 2, 3 }; + Span a3 = stackalloc [] { 1, 2, 3 }; + } + } +}").VerifyDiagnostics( + // (11,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' + // Span a1 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [3] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(11, 28), + // (12,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' + // Span a2 = stackalloc int [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(12, 28), + // (13,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' + // Span a3 = stackalloc [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc [] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(13, 28) + ); + } + + [Fact] + public void ConditionalExpressionOnSpan_BothStackallocSpans() + { + CreateCompilationWithMscorlibAndSpan(@" +class Test +{ + void M() + { + var x1 = true ? stackalloc int [3] { 1, 2, 3, } : stackalloc int [3] { 1, 2, 3, }; + var x2 = true ? stackalloc int [] { 1, 2, 3, } : stackalloc int [] { 1, 2, 3, }; + var x3 = true ? stackalloc [] { 1, 2, 3, } : stackalloc [] { 1, 2, 3, }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + } + + [Fact] + public void ConditionalExpressionOnSpan_Convertible() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + void M() + { + var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc int [3] { 1, 2, 3, }; + var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc int [] { 1, 2, 3, }; + var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { 1, 2, 3, }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + } + + [Fact] + public void ConditionalExpressionOnSpan_NoCast() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + void M() + { + var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc short [3] { (short)1, (short)2, (short)3 }; + var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc short [] { (short)1, (short)2, (short)3 }; + var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { (short)1, (short)2, (short)3 }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (7,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. + // var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc short [3] { (short)1, (short)2, (short)3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc short [3] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(7, 59), + // (8,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. + // var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc short [] { (short)1, (short)2, (short)3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc short [] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(8, 59), + // (9,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. + // var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { (short)1, (short)2, (short)3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc [] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(9, 59) + ); + } + + [Fact] + public void ConditionalExpressionOnSpan_CompatibleTypes() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + void M() + { + Span a = stackalloc int [10]; + var x = true ? stackalloc int [10] : a; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + } + + [Fact] + public void ConditionalExpressionOnSpan_IncompatibleTypes() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + void M() + { + Span a = stackalloc short [10]; + var x1 = true ? stackalloc int [3] { 1, 2, 3 } : a; + var x2 = true ? stackalloc int [] { 1, 2, 3 } : a; + var x3 = true ? stackalloc [] { 1, 2, 3 } : a; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (8,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[3]' and 'Span' + // var x1 = true ? stackalloc int [3] { 1, 2, 3 } : a; + Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc int [3] { 1, 2, 3 } : a").WithArguments("stackalloc int[3]", "System.Span").WithLocation(8, 18), + // (9,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[stackalloc int [] { 1, 2, 3 }]' and 'Span' + // var x2 = true ? stackalloc int [] { 1, 2, 3 } : a; + Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc int [] { 1, 2, 3 } : a").WithArguments("stackalloc int[stackalloc int [] { 1, 2, 3 }]", "System.Span").WithLocation(9, 18), + // (10,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[stackalloc [] { 1, 2, 3 }]' and 'Span' + // var x3 = true ? stackalloc [] { 1, 2, 3 } : a; + Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc [] { 1, 2, 3 } : a").WithArguments("stackalloc int[stackalloc [] { 1, 2, 3 }]", "System.Span").WithLocation(10, 18) + ); + } + + [Fact] + public void ConditionalExpressionOnSpan_Nested() + { + CreateCompilationWithMscorlibAndSpan(@" +class Test +{ + bool N() => true; + + void M() + { + var x = N() + ? N() + ? stackalloc int [1] { 2 } + : stackalloc int[] { 2 } + : N() + ? stackalloc[] { 2 } + : N() + ? stackalloc int[4] + : stackalloc int[5]; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + } + + [Fact] + public void BooleanOperatorOnSpan_NoTargetTyping() + { + CreateCompilationWithMscorlibAndSpan(@" +class Test +{ + void M() + { + if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } + if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } + if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (6,12): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 12), + // (6,39): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 39), + // (7,12): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 12), + // (7,39): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 39), + // (8,12): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 12), + // (8,39): error CS1525: Invalid expression term 'stackalloc' + // if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 39) + ); + } + + [Fact] + public void StackAllocInitializerSyntaxProducesErrorsOnEarlierVersions() + { + var parseOptions = new CSharpParseOptions().WithLanguageVersion(LanguageVersion.CSharp7); + + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + void M() + { + Span x1 = stackalloc int[1] { 2 }; + Span x2 = stackalloc int[] { 2 }; + Span x3 = stackalloc [] { 2 }; + } +}", options: TestOptions.UnsafeReleaseDll, parseOptions: parseOptions).VerifyDiagnostics( + // (7,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. + // Span x1 = stackalloc int[1] { 2 }; + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(7, 24), + // (8,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. + // Span x2 = stackalloc int[] { 2 }; + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(8, 24), + // (9,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. + // Span x3 = stackalloc [] { 2 }; + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(9, 24) + ); + } + + [Fact] + public void StackAllocSyntaxProducesUnsafeErrorInSafeCode() + { + CreateStandardCompilation(@" +class Test +{ + void M() + { + var x1 = stackalloc int[1] { 2 }; + var x2 = stackalloc int[] { 2 }; + var x3 = stackalloc [] { 2 }; + } +}", options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (6,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var x1 = stackalloc int[1] { 2 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[1] { 2 }").WithLocation(6, 18), + // (7,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var x2 = stackalloc int[] { 2 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[] { 2 }").WithLocation(7, 18), + // (8,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var x3 = stackalloc [] { 2 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc [] { 2 }").WithLocation(8, 18) + ); + } + + [Fact] + public void StackAllocInUsing1() + { + var test = @" +public class Test +{ + unsafe public static void Main() + { + using (var v1 = stackalloc int[1] { 2 } ) + using (var v2 = stackalloc int[] { 2 } ) + using (var v3 = stackalloc [] { 2 } ) + { + } + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (6,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' + // using (var v1 = stackalloc int[1] { 2 } ) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v1 = stackalloc int[1] { 2 }").WithArguments("int*").WithLocation(6, 16), + // (7,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' + // using (var v2 = stackalloc int[] { 2 } ) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v2 = stackalloc int[] { 2 }").WithArguments("int*").WithLocation(7, 16), + // (8,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' + // using (var v3 = stackalloc [] { 2 } ) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v3 = stackalloc [] { 2 }").WithArguments("int*").WithLocation(8, 16) + ); + } + + [Fact] + public void StackAllocInUsing2() + { + var test = @" +public class Test +{ + unsafe public static void Main() + { + using (System.IDisposable v1 = stackalloc int[1] { 2 } ) + using (System.IDisposable v2 = stackalloc int[] { 2 } ) + using (System.IDisposable v3 = stackalloc [] { 2 } ) + { + } + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (6,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. + // using (System.IDisposable v1 = stackalloc int[1] { 2 } ) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[1] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(6, 40), + // (7,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. + // using (System.IDisposable v2 = stackalloc int[] { 2 } ) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(7, 40), + // (8,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. + // using (System.IDisposable v3 = stackalloc [] { 2 } ) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc [] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(8, 40) + ); + } + + [Fact] + public void ConstStackAllocExpression() + { + var test = @" +unsafe public class Test +{ + void M() + { + const int* p1 = stackalloc int[1] { 1 }; + const int* p2 = stackalloc int[] { 1 }; + const int* p3 = stackalloc [] { 1 }; + } +} +"; + CreateStandardCompilation(test, options: new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary, allowUnsafe: true)).VerifyDiagnostics( + // (6,15): error CS0283: The type 'int*' cannot be declared const + // const int* p1 = stackalloc int[1] { 1 }; + Diagnostic(ErrorCode.ERR_BadConstType, "int*").WithArguments("int*").WithLocation(6, 15), + // (7,15): error CS0283: The type 'int*' cannot be declared const + // const int* p2 = stackalloc int[] { 1 }; + Diagnostic(ErrorCode.ERR_BadConstType, "int*").WithArguments("int*").WithLocation(7, 15), + // (8,15): error CS0283: The type 'int*' cannot be declared const + // const int* p3 = stackalloc [] { 1 }; + Diagnostic(ErrorCode.ERR_BadConstType, "int*").WithArguments("int*").WithLocation(8, 15) + ); + } + + [Fact] + public void RefStackAllocAssignment_ValueToRef() + { + var test = @" +using System; +public class Test +{ + void M() + { + ref Span p1 = stackalloc int[1] { 1 }; + ref Span p2 = stackalloc int[] { 1 }; + ref Span p3 = stackalloc [] { 1 }; + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (7,23): error CS8172: Cannot initialize a by-reference variable with a value + // ref Span p1 = stackalloc int[1] { 1 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p1 = stackalloc int[1] { 1 }").WithLocation(7, 23), + // (7,28): error CS1510: A ref or out value must be an assignable variable + // ref Span p1 = stackalloc int[1] { 1 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int[1] { 1 }").WithLocation(7, 28), + // (8,23): error CS8172: Cannot initialize a by-reference variable with a value + // ref Span p2 = stackalloc int[] { 1 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p2 = stackalloc int[] { 1 }").WithLocation(8, 23), + // (8,28): error CS1510: A ref or out value must be an assignable variable + // ref Span p2 = stackalloc int[] { 1 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int[] { 1 }").WithLocation(8, 28), + // (9,23): error CS8172: Cannot initialize a by-reference variable with a value + // ref Span p3 = stackalloc [] { 1 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p3 = stackalloc [] { 1 }").WithLocation(9, 23), + // (9,28): error CS1510: A ref or out value must be an assignable variable + // ref Span p3 = stackalloc [] { 1 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc [] { 1 }").WithLocation(9, 28) + ); + } + + [Fact] + public void RefStackAllocAssignment_RefToRef() + { + var test = @" +using System; +public class Test +{ + void M() + { + ref Span p1 = ref stackalloc int [3] { 1, 2, 3 }; + ref Span p2 = ref stackalloc int [] { 1, 2, 3 }; + ref Span p3 = ref stackalloc [] { 1, 2, 3 }; + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (7,32): error CS1525: Invalid expression term 'stackalloc' + // ref Span p1 = ref stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 32), + // (8,32): error CS1525: Invalid expression term 'stackalloc' + // ref Span p2 = ref stackalloc int [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 32), + // (9,32): error CS1525: Invalid expression term 'stackalloc' + // ref Span p3 = ref stackalloc [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(9, 32) + ); + } + + [Fact] + public void InvalidPositionForStackAllocSpan() + { + var test = @" +using System; +public class Test +{ + void M() + { + N(stackalloc int [3] { 1, 2, 3 }); + N(stackalloc int [] { 1, 2, 3 }); + N(stackalloc [] { 1, 2, 3 }); + } + void N(Span span) + { + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (7,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc int [3] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 11), + // (8,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc int [] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 11), + // (9,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc [] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(9, 11) + ); + } + + [Fact] + public void CannotDotIntoStackAllocExpression() + { + var test = @" +public class Test +{ + void M() + { + int length1 = (stackalloc int [3] { 1, 2, 3 }).Length; + int length2 = (stackalloc int [] { 1, 2, 3 }).Length; + int length3 = (stackalloc [] { 1, 2, 3 }).Length; + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, TestOptions.ReleaseDll).VerifyDiagnostics( + // (6,24): error CS1525: Invalid expression term 'stackalloc' + // int length1 = (stackalloc int [3] { 1, 2, 3 }).Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 24), + // (7,24): error CS1525: Invalid expression term 'stackalloc' + // int length2 = (stackalloc int [] { 1, 2, 3 }).Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 24), + // (8,24): error CS1525: Invalid expression term 'stackalloc' + // int length3 = (stackalloc [] { 1, 2, 3 }).Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 24) + ); + } + + [Fact] + public void OverloadResolution_Fail() + { + var test = @" +using System; +unsafe public class Test +{ + static void Main() + { + Invoke(stackalloc int[3] { 1, 2, 3 }); + Invoke(stackalloc int[] { 1, 2, 3 }); + Invoke(stackalloc[] { 1, 2, 3 }); + } + + static void Invoke(Span shortSpan) => Console.WriteLine(""shortSpan""); + static void Invoke(Span boolSpan) => Console.WriteLine(""boolSpan""); + static void Invoke(int* intPointer) => Console.WriteLine(""intPointer""); + static void Invoke(void* voidPointer) => Console.WriteLine(""voidPointer""); +} +"; + CreateCompilationWithMscorlibAndSpan(test, TestOptions.UnsafeReleaseExe).VerifyDiagnostics( + // (7,16): error CS1525: Invalid expression term 'stackalloc' + // Invoke(stackalloc int[3] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 16), + // (8,16): error CS1525: Invalid expression term 'stackalloc' + // Invoke(stackalloc int[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 16), + // (9,16): error CS1525: Invalid expression term 'stackalloc' + // Invoke(stackalloc[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(9, 16) + ); + } + + [Fact] + public void StackAllocWithDynamic() + { + CreateStandardCompilation(@" +class Program +{ + static void Main() + { + dynamic d = 1; + var d1 = stackalloc dynamic[1] { d }; + var d2 = stackalloc dynamic[] { d }; + var d3 = stackalloc[] { d }; + } +}").VerifyDiagnostics( + // (7,29): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // var d1 = stackalloc dynamic[1] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(7, 29), + // (8,29): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // var d2 = stackalloc dynamic[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(8, 29), + // (9,18): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // var d3 = stackalloc[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(9, 18), + // (9,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var d3 = stackalloc[] { d }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc[] { d }").WithLocation(9, 18) + ); + } + + [Fact] + public void StackAllocWithDynamicSpan() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Program +{ + static void Main() + { + dynamic d = 1; + Span d1 = stackalloc dynamic[1] { d }; + Span d2 = stackalloc dynamic[] { d }; + Span d3 = stackalloc[] { d }; + } +}").VerifyDiagnostics( + // (8,39): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // Span d1 = stackalloc dynamic[1] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(8, 39), + // (9,39): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // Span d2 = stackalloc dynamic[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(9, 39), + // (10,28): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // Span d3 = stackalloc[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(10, 28) + ); + } + + [Fact] + public void StackAllocAsArgument() + { + CreateStandardCompilation(@" +class Program +{ + static void N(object p) { } + + static void Main() + { + N(stackalloc int[3] { 1, 2, 3 }); + N(stackalloc int[] { 1, 2, 3 }); + N(stackalloc[] { 1, 2, 3 }); + } +}").VerifyDiagnostics( + // (8,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc int[3] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 11), + // (9,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc int[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(9, 11), + // (10,11): error CS1525: Invalid expression term 'stackalloc' + // N(stackalloc[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(10, 11) + ); + } + + [Fact] + public void StackAllocInParenthesis() + { + CreateStandardCompilation(@" +class Program +{ + static void Main() + { + var x1 = (stackalloc int[3] { 1, 2, 3 }); + var x2 = (stackalloc int[] { 1, 2, 3 }); + var x3 = (stackalloc[] { 1, 2, 3 }); + } +}").VerifyDiagnostics( + // (6,19): error CS1525: Invalid expression term 'stackalloc' + // var x1 = (stackalloc int[3] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 19), + // (7,19): error CS1525: Invalid expression term 'stackalloc' + // var x2 = (stackalloc int[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 19), + // (8,19): error CS1525: Invalid expression term 'stackalloc' + // var x3 = (stackalloc[] { 1, 2, 3 }); + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 19) + ); + } + + [Fact] + public void StackAllocInNullConditionalOperator() + { + CreateStandardCompilation(@" +class Program +{ + static void Main() + { + var x1 = stackalloc int [3] { 1, 2, 3 } ?? stackalloc int [3] { 1, 2, 3 }; + var x2 = stackalloc int [] { 1, 2, 3 } ?? stackalloc int [] { 1, 2, 3 }; + var x3 = stackalloc [] { 1, 2, 3 } ?? stackalloc [] { 1, 2, 3 }; + } +}").VerifyDiagnostics( + // (6,18): error CS1525: Invalid expression term 'stackalloc' + // var x1 = stackalloc int [3] { 1, 2, 3 } ?? stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 18), + // (6,52): error CS1525: Invalid expression term 'stackalloc' + // var x1 = stackalloc int [3] { 1, 2, 3 } ?? stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 52), + // (7,18): error CS1525: Invalid expression term 'stackalloc' + // var x2 = stackalloc int [] { 1, 2, 3 } ?? stackalloc int [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 18), + // (7,52): error CS1525: Invalid expression term 'stackalloc' + // var x2 = stackalloc int [] { 1, 2, 3 } ?? stackalloc int [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 52), + // (8,18): error CS1525: Invalid expression term 'stackalloc' + // var x3 = stackalloc [] { 1, 2, 3 } ?? stackalloc [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 18), + // (8,52): error CS1525: Invalid expression term 'stackalloc' + // var x3 = stackalloc [] { 1, 2, 3 } ?? stackalloc [] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 52) + ); + } + + [Fact] + public void StackAllocInCastAndConditionalOperator() + { + CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + public void Method() + { + Test value1 = true ? new Test() : (Test)stackalloc int[3] { 1, 2, 3 }; + Test value2 = true ? new Test() : (Test)stackalloc int[] { 1, 2, 3 }; + Test value3 = true ? new Test() : (Test)stackalloc[] { 1, 2, 3 }; + } + + public static explicit operator Test(Span value) + { + return new Test(); + } +}", TestOptions.ReleaseDll).VerifyDiagnostics(); + } + } + +} diff --git a/src/Compilers/CSharp/Test/Syntax/Generated/Syntax.Test.xml.Generated.cs b/src/Compilers/CSharp/Test/Syntax/Generated/Syntax.Test.xml.Generated.cs index 1a1a9507640c8..8f73b5bd2dc0b 100644 --- a/src/Compilers/CSharp/Test/Syntax/Generated/Syntax.Test.xml.Generated.cs +++ b/src/Compilers/CSharp/Test/Syntax/Generated/Syntax.Test.xml.Generated.cs @@ -286,7 +286,12 @@ private static Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.ImplicitArray private static Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.StackAllocArrayCreationExpressionSyntax GenerateStackAllocArrayCreationExpression() { - return Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.StackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), GenerateIdentifierName()); + return Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.StackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), GenerateIdentifierName(), null); + } + + private static Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.ImplicitStackAllocArrayCreationExpressionSyntax GenerateImplicitStackAllocArrayCreationExpression() + { + return Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.ImplicitStackAllocArrayCreationExpression(Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.Token(SyntaxKind.OpenBracketToken), Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.SyntaxFactory.Token(SyntaxKind.CloseBracketToken), GenerateInitializerExpression()); } private static Microsoft.CodeAnalysis.CSharp.Syntax.InternalSyntax.QueryExpressionSyntax GenerateQueryExpression() @@ -1686,6 +1691,20 @@ public void TestStackAllocArrayCreationExpressionFactoryAndProperties() Assert.Equal(SyntaxKind.StackAllocKeyword, node.StackAllocKeyword.Kind); Assert.NotNull(node.Type); + Assert.Null(node.Initializer); + + AttachAndCheckDiagnostics(node); + } + + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionFactoryAndProperties() + { + var node = GenerateImplicitStackAllocArrayCreationExpression(); + + Assert.Equal(SyntaxKind.StackAllocKeyword, node.StackAllocKeyword.Kind); + Assert.Equal(SyntaxKind.OpenBracketToken, node.OpenBracketToken.Kind); + Assert.Equal(SyntaxKind.CloseBracketToken, node.CloseBracketToken.Kind); + Assert.NotNull(node.Initializer); AttachAndCheckDiagnostics(node); } @@ -5075,6 +5094,32 @@ public void TestStackAllocArrayCreationExpressionIdentityRewriter() Assert.Same(oldNode, newNode); } + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionTokenDeleteRewriter() + { + var oldNode = GenerateImplicitStackAllocArrayCreationExpression(); + var rewriter = new TokenDeleteRewriter(); + var newNode = rewriter.Visit(oldNode); + + if(!oldNode.IsMissing) + { + Assert.NotEqual(oldNode, newNode); + } + + Assert.NotNull(newNode); + Assert.True(newNode.IsMissing, "No tokens => missing"); + } + + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionIdentityRewriter() + { + var oldNode = GenerateImplicitStackAllocArrayCreationExpression(); + var rewriter = new IdentityRewriter(); + var newNode = rewriter.Visit(oldNode); + + Assert.Same(oldNode, newNode); + } + [Fact] public void TestQueryExpressionTokenDeleteRewriter() { @@ -9205,7 +9250,12 @@ private static ImplicitArrayCreationExpressionSyntax GenerateImplicitArrayCreati private static StackAllocArrayCreationExpressionSyntax GenerateStackAllocArrayCreationExpression() { - return SyntaxFactory.StackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), GenerateIdentifierName()); + return SyntaxFactory.StackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), GenerateIdentifierName(), default(InitializerExpressionSyntax)); + } + + private static ImplicitStackAllocArrayCreationExpressionSyntax GenerateImplicitStackAllocArrayCreationExpression() + { + return SyntaxFactory.ImplicitStackAllocArrayCreationExpression(SyntaxFactory.Token(SyntaxKind.StackAllocKeyword), SyntaxFactory.Token(SyntaxKind.OpenBracketToken), SyntaxFactory.Token(SyntaxKind.CloseBracketToken), GenerateInitializerExpression()); } private static QueryExpressionSyntax GenerateQueryExpression() @@ -10605,7 +10655,21 @@ public void TestStackAllocArrayCreationExpressionFactoryAndProperties() Assert.Equal(SyntaxKind.StackAllocKeyword, node.StackAllocKeyword.Kind()); Assert.NotNull(node.Type); - var newNode = node.WithStackAllocKeyword(node.StackAllocKeyword).WithType(node.Type); + Assert.Null(node.Initializer); + var newNode = node.WithStackAllocKeyword(node.StackAllocKeyword).WithType(node.Type).WithInitializer(node.Initializer); + Assert.Equal(node, newNode); + } + + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionFactoryAndProperties() + { + var node = GenerateImplicitStackAllocArrayCreationExpression(); + + Assert.Equal(SyntaxKind.StackAllocKeyword, node.StackAllocKeyword.Kind()); + Assert.Equal(SyntaxKind.OpenBracketToken, node.OpenBracketToken.Kind()); + Assert.Equal(SyntaxKind.CloseBracketToken, node.CloseBracketToken.Kind()); + Assert.NotNull(node.Initializer); + var newNode = node.WithStackAllocKeyword(node.StackAllocKeyword).WithOpenBracketToken(node.OpenBracketToken).WithCloseBracketToken(node.CloseBracketToken).WithInitializer(node.Initializer); Assert.Equal(node, newNode); } @@ -13994,6 +14058,32 @@ public void TestStackAllocArrayCreationExpressionIdentityRewriter() Assert.Same(oldNode, newNode); } + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionTokenDeleteRewriter() + { + var oldNode = GenerateImplicitStackAllocArrayCreationExpression(); + var rewriter = new TokenDeleteRewriter(); + var newNode = rewriter.Visit(oldNode); + + if(!oldNode.IsMissing) + { + Assert.NotEqual(oldNode, newNode); + } + + Assert.NotNull(newNode); + Assert.True(newNode.IsMissing, "No tokens => missing"); + } + + [Fact] + public void TestImplicitStackAllocArrayCreationExpressionIdentityRewriter() + { + var oldNode = GenerateImplicitStackAllocArrayCreationExpression(); + var rewriter = new IdentityRewriter(); + var newNode = rewriter.Visit(oldNode); + + Assert.Same(oldNode, newNode); + } + [Fact] public void TestQueryExpressionTokenDeleteRewriter() { diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs index 0615e1f626f34..3cace8bc0a95b 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs @@ -3,6 +3,7 @@ using System.Linq; using Microsoft.CodeAnalysis.CSharp.Syntax; using Microsoft.CodeAnalysis.CSharp.Test.Utilities; +using Microsoft.CodeAnalysis.Test.Utilities; using Roslyn.Test.Utilities; using Xunit; using Xunit.Abstractions; diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/ParsingTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/ParsingTests.cs index e9d3ac13b71fd..4b492376cbad9 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/ParsingTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/ParsingTests.cs @@ -62,9 +62,9 @@ internal void UsingStatement(string text, params DiagnosticDescription[] expecte UsingNode(node); } - internal void UsingExpression(string text, params DiagnosticDescription[] expectedErrors) + internal void UsingExpression(string text, ParseOptions options, params DiagnosticDescription[] expectedErrors) { - var node = SyntaxFactory.ParseExpression(text); + var node = SyntaxFactory.ParseExpression(text, options: options); // we validate the text roundtrips Assert.Equal(text, node.ToFullString()); var actualErrors = node.GetDiagnostics(); @@ -72,6 +72,11 @@ internal void UsingExpression(string text, params DiagnosticDescription[] expect UsingNode(node); } + internal void UsingExpression(string text, params DiagnosticDescription[] expectedErrors) + { + UsingExpression(text, options: null, expectedErrors); + } + /// /// Parses given string and initializes a depth-first preorder enumerator. /// diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs new file mode 100644 index 0000000000000..62036f4f4dd98 --- /dev/null +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs @@ -0,0 +1,113 @@ +// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using Xunit; +using Microsoft.CodeAnalysis.CSharp.Test.Utilities; +using Microsoft.CodeAnalysis.Test.Utilities; +using Xunit.Abstractions; + +namespace Microsoft.CodeAnalysis.CSharp.UnitTests.Parsing +{ + [CompilerTrait(CompilerFeature.StackAllocInitializer)] + public class StackAllocInitializerTests : ParsingTests + { + public StackAllocInitializerTests(ITestOutputHelper output) : base(output) { } + + [Fact] + public void StackAllocInitializer_01() + { + UsingExpression("stackalloc int[] { 42 }", options: TestOptions.Regular7, + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + N(SyntaxKind.StackAllocArrayCreationExpression); + { + N(SyntaxKind.StackAllocKeyword); + N(SyntaxKind.ArrayType); + { + N(SyntaxKind.PredefinedType); + { + N(SyntaxKind.IntKeyword); + } + N(SyntaxKind.ArrayRankSpecifier); + { + N(SyntaxKind.OpenBracketToken); + N(SyntaxKind.OmittedArraySizeExpression); + { + N(SyntaxKind.OmittedArraySizeExpressionToken); + } + N(SyntaxKind.CloseBracketToken); + } + } + N(SyntaxKind.ArrayInitializerExpression); + { + N(SyntaxKind.OpenBraceToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "42"); + } + N(SyntaxKind.CloseBraceToken); + } + } + EOF(); + + } + + [Fact] + public void StackAllocInitializer_02() + { + UsingExpression("stackalloc int[1] { 42 }", options: TestOptions.Regular7, + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + N(SyntaxKind.StackAllocArrayCreationExpression); + { + N(SyntaxKind.StackAllocKeyword); + N(SyntaxKind.ArrayType); + { + N(SyntaxKind.PredefinedType); + { + N(SyntaxKind.IntKeyword); + } + N(SyntaxKind.ArrayRankSpecifier); + { + N(SyntaxKind.OpenBracketToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "1"); + } + N(SyntaxKind.CloseBracketToken); + } + } + N(SyntaxKind.ArrayInitializerExpression); + { + N(SyntaxKind.OpenBraceToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "42"); + } + N(SyntaxKind.CloseBraceToken); + } + } + EOF(); + } + + [Fact] + public void StackAllocInitializer_03() + { + UsingExpression("stackalloc[] { 42 }", options: TestOptions.Regular7, + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + N(SyntaxKind.ImplicitStackAllocArrayCreationExpression); + { + N(SyntaxKind.StackAllocKeyword); + N(SyntaxKind.OpenBracketToken); + N(SyntaxKind.CloseBracketToken); + N(SyntaxKind.ArrayInitializerExpression); + { + N(SyntaxKind.OpenBraceToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "42"); + } + N(SyntaxKind.CloseBraceToken); + } + } + EOF(); + } + } +} diff --git a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs index 761c8a7135082..ae4fc259037e0 100644 --- a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs +++ b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs @@ -4,6 +4,7 @@ using System.Collections.Generic; using System.Collections.Immutable; using System.Diagnostics; +using System.Linq; using System.Reflection.Metadata; using Roslyn.Utilities; @@ -87,6 +88,29 @@ internal void EmitArrayBlockInitializer(ImmutableArray data, SyntaxNode sy EmitToken(initializeArray, syntaxNode, diagnostics); } + internal void EmitStackAllocBlockInitializer(ImmutableArray data, SyntaxNode syntaxNode, DiagnosticBag diagnostics) + { + if (data.All(datum => datum == data[0])) + { + // All bytes are the same, no need for metadata blob + + EmitOpCode(ILOpCode.Dup); + EmitIntConstant(data[0]); + EmitIntConstant(data.Length); + EmitOpCode(ILOpCode.Initblk, -3); + } + else + { + var field = module.GetFieldForData(data, syntaxNode, diagnostics); + + EmitOpCode(ILOpCode.Dup); + EmitOpCode(ILOpCode.Ldsflda); + EmitToken(field, syntaxNode, diagnostics); + EmitIntConstant(data.Length); + EmitOpCode(ILOpCode.Cpblk, -3); + } + } + /// /// Mark current IL position with a label /// diff --git a/src/Test/Utilities/Portable/TestResource.resx b/src/Test/Utilities/Portable/TestResource.resx index b7f0f026124bb..f33c01a9193e2 100644 --- a/src/Test/Utilities/Portable/TestResource.resx +++ b/src/Test/Utilities/Portable/TestResource.resx @@ -386,7 +386,7 @@ namespace My { *intref = 1; } - fixed (int* p = stackalloc int[100]) + fixed (int* p = stackalloc int[] { 100 }, q = stackalloc[] { 100 }) { *intref = 1; } diff --git a/src/Test/Utilities/Portable/Traits/CompilerFeature.cs b/src/Test/Utilities/Portable/Traits/CompilerFeature.cs index 8d176ca617c63..58b67f351778d 100644 --- a/src/Test/Utilities/Portable/Traits/CompilerFeature.cs +++ b/src/Test/Utilities/Portable/Traits/CompilerFeature.cs @@ -28,5 +28,6 @@ public enum CompilerFeature PrivateProtected, PEVerifyCompat, RefConditionalOperator, + StackAllocInitializer, } } From 1ff896d5aa07ae5715aae70b126f43694d47036b Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 26 Jan 2018 03:49:19 +0330 Subject: [PATCH 02/43] Update resources --- src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf | 5 +++++ src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf | 5 +++++ .../CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf | 5 +++++ .../CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf | 5 +++++ 13 files changed, 65 insertions(+) diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf index 295b41afcae2e..9f9aa698dd242 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf @@ -8610,6 +8610,11 @@ Pokud chcete odstranit toto varování, můžete místo toho použít /reference ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf index f83a782a786fa..8552a32a89502 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf @@ -8610,6 +8610,11 @@ Um die Warnung zu beheben, können Sie stattdessen /reference verwenden (Einbett ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf index 8d4feb9eacc7e..6c7b8cb95e918 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf @@ -8610,6 +8610,11 @@ Para eliminar la advertencia puede usar /reference (establezca la propiedad Embe ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf index 4bd455eb0860a..4f2227605ac2d 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf @@ -8610,6 +8610,11 @@ Pour supprimer l'avertissement, vous pouvez utiliser la commande /reference (dé ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf index 6b71e3ed4adcf..9145240bd7dfd 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf @@ -8610,6 +8610,11 @@ Per rimuovere l'avviso, è invece possibile usare /reference (impostare la propr ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf index b0ba89a18fde3..11c69dadec18f 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf @@ -8610,6 +8610,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf index 5964ba1d21d98..45b0457798ae0 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf @@ -8610,6 +8610,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf index 3fa8d88eb9e3c..c80d8402301ee 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf @@ -8610,6 +8610,11 @@ Aby usunąć ostrzeżenie, możesz zamiast tego użyć opcji /reference (ustaw w ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf index 8da7eec0a3595..7ad7af67c55a1 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf @@ -8610,6 +8610,11 @@ Para incorporar informações de tipo de interoperabilidade para os dois assembl ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf index 395bfa09caa9e..b7de0efd1d989 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf @@ -8610,6 +8610,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf index d0f8bdb24ce06..28d9774b9e13b 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf @@ -8610,6 +8610,11 @@ Uyarıyı kaldırmak için, /reference kullanabilirsiniz (Birlikte Çalışma T ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf index a6567020b924a..e2b2f8762aeaf 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf @@ -8610,6 +8610,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf index 862c51a1a77e7..b434983a3054b 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf @@ -8610,6 +8610,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ ref conditional expression + + stackalloc initilizer + stackalloc initilizer + + \ No newline at end of file From edc09fc50f0820f949b69f218179ee61542aea37 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 26 Jan 2018 11:23:14 +0330 Subject: [PATCH 03/43] Address test failures --- .../Portable/Binder/Binder_Expressions.cs | 8 ++-- .../CSharp/Portable/Parser/LanguageParser.cs | 6 +-- .../Semantics/StackAllocInitializerTests.cs | 46 ++++++++++++++++++- .../Test/Semantic/Semantics/UnsafeTests.cs | 36 ++++++++------- 4 files changed, 71 insertions(+), 25 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index cff5dc0a2abdb..ba24133bf3414 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -2743,7 +2743,7 @@ private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitSt return BindStackAllocWithInitializer( node, initializer, - type: GetStackAllocType(node, bestType, inLegalPosition, diagnostics), + type: inLegalPosition ? GetStackAllocType(node, bestType, diagnostics) : null, elementType: bestType, sizeOpt: null, diagnostics, @@ -3132,7 +3132,7 @@ private BoundExpression BindStackAllocArrayCreationExpression( new PointerTypeSymbol(elementType)); } - TypeSymbol type = GetStackAllocType(node, elementType, inLegalPosition, diagnostics); + TypeSymbol type = inLegalPosition ? GetStackAllocType(node, elementType, diagnostics) : null; ExpressionSyntax countSyntax = rankSpecifiers[0].Sizes[0]; BoundExpression count = null; @@ -3178,9 +3178,9 @@ private bool ReportBadStackAllocPosition(SyntaxNode node, DiagnosticBag diagnost return inLegalPosition; } - private TypeSymbol GetStackAllocType(SyntaxNode node, TypeSymbol elementType, bool inLegalPosition, DiagnosticBag diagnostics) + private TypeSymbol GetStackAllocType(SyntaxNode node, TypeSymbol elementType, DiagnosticBag diagnostics) { - if (inLegalPosition && !node.IsVariableDeclarationInitialization()) + if (!node.IsVariableDeclarationInitialization()) { CheckFeatureAvailability(node, MessageID.IDS_FeatureRefStructs, diagnostics); GetWellKnownTypeMember(Compilation, WellKnownMember.System_Span_T__ctor, diagnostics, syntax: node); diff --git a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs index 4b73468bfb623..6043e070a35cb 100644 --- a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs +++ b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs @@ -10712,21 +10712,19 @@ private ExpressionSyntax ParseRegularStackAllocExpression() { var @stackalloc = this.EatToken(SyntaxKind.StackAllocKeyword); var elementType = this.ParseType(expectSizes: true); - InitializerExpressionSyntax initializer; + InitializerExpressionSyntax initializer = null; if (this.CurrentToken.Kind == SyntaxKind.OpenBraceToken) { @stackalloc = CheckFeatureAvailability(@stackalloc, MessageID.IDS_FeatureStackAllocInitializer); initializer = this.ParseArrayInitializer(); } - else + else if (elementType.Kind == SyntaxKind.ArrayType) { var rankSpec = ((ArrayTypeSyntax)elementType).RankSpecifiers[0]; if (GetNumberOfNonOmittedArraySizes(rankSpec) == 0) { elementType = this.AddError(elementType, rankSpec, ErrorCode.ERR_MissingArraySize); } - - initializer = null; } return _syntaxFactory.StackAllocArrayCreationExpression(@stackalloc, elementType, initializer); diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 7a88323d81a9e..c9d2bf7a0f6b0 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -151,6 +151,8 @@ unsafe class Test public void Method1() { var obj1 = stackalloc[] {{1}}; + var obj2 = stackalloc int[] {{1}}; + var obj3 = stackalloc int[1] {{1}}; } }", TestOptions.UnsafeReleaseDll); @@ -160,7 +162,49 @@ public void Method1() Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34), // (6,34): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. // var obj1 = stackalloc[] {{1}}; - Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34) + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34), + // (7,38): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj2 = stackalloc int[] {{1}}; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(7, 38), + // (8,39): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj3 = stackalloc int[1] {{1}}; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(8, 39) + ); + } + + [Fact] + public void AsStatement() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public void Method1() + { + stackalloc[] {1}; + stackalloc int[] {1}; + stackalloc int[1] {1}; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,9): error CS1525: Invalid expression term 'stackalloc' + // stackalloc[] {1}; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 9), + // (6,9): error CS0201: Only assignment, call, increment, decrement, and new object expressions can be used as a statement + // stackalloc[] {1}; + Diagnostic(ErrorCode.ERR_IllegalStatement, "stackalloc[] {1}").WithLocation(6, 9), + // (7,9): error CS1525: Invalid expression term 'stackalloc' + // stackalloc int[] {1}; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 9), + // (7,9): error CS0201: Only assignment, call, increment, decrement, and new object expressions can be used as a statement + // stackalloc int[] {1}; + Diagnostic(ErrorCode.ERR_IllegalStatement, "stackalloc int[] {1}").WithLocation(7, 9), + // (8,9): error CS1525: Invalid expression term 'stackalloc' + // stackalloc int[1] {1}; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 9), + // (8,9): error CS0201: Only assignment, call, increment, decrement, and new object expressions can be used as a statement + // stackalloc int[1] {1}; + Diagnostic(ErrorCode.ERR_IllegalStatement, "stackalloc int[1] {1}").WithLocation(8, 9) ); } diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/UnsafeTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/UnsafeTests.cs index 2ef4b96822138..7fb67b666385f 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/UnsafeTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/UnsafeTests.cs @@ -7463,33 +7463,37 @@ static void Main() } "; CreateStandardCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( - // (6,31): error CS1575: A stackalloc expression requires [] after type + // (6,34): error CS1586: Array creation must have array size or array initializer // { int* p = stackalloc int[]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[]"), + Diagnostic(ErrorCode.ERR_MissingArraySize, "[]").WithLocation(6, 34), + // (8,34): error CS1586: Array creation must have array size or array initializer + // { int* p = stackalloc int[][]; } + Diagnostic(ErrorCode.ERR_MissingArraySize, "[]").WithLocation(8, 34), + // (9,34): error CS1586: Array creation must have array size or array initializer + // { int* p = stackalloc int[][1]; } + Diagnostic(ErrorCode.ERR_MissingArraySize, "[]").WithLocation(9, 34), + // (9,37): error CS0270: Array size cannot be specified in a variable declaration (try initializing with a 'new' expression) + // { int* p = stackalloc int[][1]; } + Diagnostic(ErrorCode.ERR_ArraySizeInDeclaration, "1").WithLocation(9, 37), + // (11,38): error CS0270: Array size cannot be specified in a variable declaration (try initializing with a 'new' expression) + // { int* p = stackalloc int[1][1]; } + Diagnostic(ErrorCode.ERR_ArraySizeInDeclaration, "1").WithLocation(11, 38), // (7,31): error CS1575: A stackalloc expression requires [] after type // { int* p = stackalloc int[1, 1]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1, 1]"), + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1, 1]").WithLocation(7, 31), // (8,31): error CS1575: A stackalloc expression requires [] after type // { int* p = stackalloc int[][]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[][]"), + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[][]").WithLocation(8, 31), // (9,31): error CS1575: A stackalloc expression requires [] after type // { int* p = stackalloc int[][1]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[][1]"), + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[][1]").WithLocation(9, 31), // (10,31): error CS1575: A stackalloc expression requires [] after type // { int* p = stackalloc int[1][]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1][]"), + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1][]").WithLocation(10, 31), // (11,31): error CS1575: A stackalloc expression requires [] after type // { int* p = stackalloc int[1][1]; } - Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1][1]"), - - // CONSIDER: these are plausible, but not ideal. - - // (9,37): error CS0270: Array size cannot be specified in a variable declaration (try initializing with a 'new' expression) - // { int* p = stackalloc int[][1]; } - Diagnostic(ErrorCode.ERR_ArraySizeInDeclaration, "1"), - // (11,38): error CS0270: Array size cannot be specified in a variable declaration (try initializing with a 'new' expression) - // { int* p = stackalloc int[1][1]; } - Diagnostic(ErrorCode.ERR_ArraySizeInDeclaration, "1")); + Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[1][1]").WithLocation(11, 31) + ); } [Fact] From cf00fbe8abc288afa41e2c4a9173af769852583c Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Wed, 31 Jan 2018 23:47:39 +0330 Subject: [PATCH 04/43] Visit initializer elements in flow analysis --- .../FlowAnalysis/PreciseAbstractFlowPass.cs | 10 ++++++- .../FlowAnalysis/RegionAnalysisTests.cs | 26 +++++++++++++++++++ 2 files changed, 35 insertions(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs b/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs index e6f14fe27664b..a3986c4091ce6 100644 --- a/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs +++ b/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs @@ -2576,12 +2576,20 @@ public override BoundNode VisitSizeOfOperator(BoundSizeOfOperator node) public override BoundNode VisitStackAllocArrayCreation(BoundStackAllocArrayCreation node) { VisitRvalue(node.Count); + + if (node.InitializerOpt != null && !node.InitializerOpt.Initializers.IsDefault) + { + foreach (var element in node.InitializerOpt.Initializers) + VisitRvalue(element); + } + + if (_trackExceptions) NotePossibleException(node); return null; } public override BoundNode VisitConvertedStackAllocExpression(BoundConvertedStackAllocExpression node) { - VisitRvalue(node.Count); + VisitStackAllocArrayCreation(node); return null; } diff --git a/src/Compilers/CSharp/Test/Semantic/FlowAnalysis/RegionAnalysisTests.cs b/src/Compilers/CSharp/Test/Semantic/FlowAnalysis/RegionAnalysisTests.cs index 48ad6cc4cd9db..cb18000e24e4e 100644 --- a/src/Compilers/CSharp/Test/Semantic/FlowAnalysis/RegionAnalysisTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/FlowAnalysis/RegionAnalysisTests.cs @@ -1150,6 +1150,32 @@ static void Main() Assert.Equal("y, x", GetSymbolNamesJoined(dataFlowAnalysisResults.WrittenOutside)); } + [Fact] + public void TestStackAllocArrayInitializer() + { + var dataFlowAnalysisResults = CompileAndAnalyzeDataFlowExpression(@" +class C { + static void Main() + { + int y = 1; + var x = stackalloc[] { +/**/ +y +/**/ + }; + } +} +"); + Assert.Equal(null, GetSymbolNamesJoined(dataFlowAnalysisResults.VariablesDeclared)); + Assert.Equal(null, GetSymbolNamesJoined(dataFlowAnalysisResults.AlwaysAssigned)); + Assert.Equal("y", GetSymbolNamesJoined(dataFlowAnalysisResults.DataFlowsIn)); + Assert.Equal(null, GetSymbolNamesJoined(dataFlowAnalysisResults.DataFlowsOut)); + Assert.Equal("y", GetSymbolNamesJoined(dataFlowAnalysisResults.ReadInside)); + Assert.Equal(null, GetSymbolNamesJoined(dataFlowAnalysisResults.ReadOutside)); + Assert.Equal(null, GetSymbolNamesJoined(dataFlowAnalysisResults.WrittenInside)); + Assert.Equal("y, x", GetSymbolNamesJoined(dataFlowAnalysisResults.WrittenOutside)); + } + [WorkItem(539286, "http://vstfdevdiv:8080/DevDiv2/DevDiv/_workitems/edit/539286")] [Fact] public void TestAnalysisInFieldInitializers() From 9e86f0612131ce936b111c2ff7420e146b3b6bc8 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 18:18:32 +0330 Subject: [PATCH 05/43] Whitespaces. --- src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs | 4 +--- src/Compilers/CSharp/Portable/Errors/MessageID.cs | 1 - .../CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs | 1 - 3 files changed, 1 insertion(+), 5 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index ba24133bf3414..d521d0b94f661 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -2716,7 +2716,6 @@ private BoundExpression BindImplicitArrayCreationExpression(ImplicitArrayCreatio sizes: ImmutableArray.Empty, boundInitExprOpt: boundInitializerExpressions); } - private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node, DiagnosticBag diagnostics) { bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); @@ -3205,13 +3204,12 @@ private BoundExpression BindStackAllocWithInitializer( bool hasErrors, ImmutableArray boundInitExprOpt = default) { - if (boundInitExprOpt.IsDefault) { boundInitExprOpt = BindArrayInitializerExpressions(initSyntax, diagnostics, dimension: 1, rank: 1) .SelectAsArray((expr, t) => GenerateConversionForAssignment(t.elementType, expr, t.diagnostics), (elementType, diagnostics)); } - + if (sizeOpt != null) { int? constantSizeOpt = GetIntegerConstantForArraySize(sizeOpt); diff --git a/src/Compilers/CSharp/Portable/Errors/MessageID.cs b/src/Compilers/CSharp/Portable/Errors/MessageID.cs index 30883e594ee18..77aaf796b9a1e 100644 --- a/src/Compilers/CSharp/Portable/Errors/MessageID.cs +++ b/src/Compilers/CSharp/Portable/Errors/MessageID.cs @@ -148,7 +148,6 @@ internal enum MessageID IDS_FeatureRefConditional = MessageBase + 12731, IDS_FeatureAttributesOnBackingFields = MessageBase + 12732, - IDS_FeatureStackAllocInitializer = MessageBase + 12733, } diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs index 62036f4f4dd98..b6a9d915153ab 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs @@ -47,7 +47,6 @@ public void StackAllocInitializer_01() } } EOF(); - } [Fact] From 2a65a3d8f00df8a1b6d23dfe8b4c28190d568f70 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:30:33 +0330 Subject: [PATCH 06/43] Explicitly import ImmutableArray extensions --- src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs index ae4fc259037e0..e367e730aae66 100644 --- a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs +++ b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs @@ -4,9 +4,9 @@ using System.Collections.Generic; using System.Collections.Immutable; using System.Diagnostics; -using System.Linq; using System.Reflection.Metadata; using Roslyn.Utilities; +using static System.Linq.ImmutableArrayExtensions; namespace Microsoft.CodeAnalysis.CodeGen { From 27253fadec4f7795d592a77cc43f7ed1763a0dff Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:30:54 +0330 Subject: [PATCH 07/43] Revert code. --- .../LocalRewriter/LocalRewriter_PointerElementAccess.cs | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs index 2434ab11f1d6a..dc51684f6a1d7 100644 --- a/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs +++ b/src/Compilers/CSharp/Portable/Lowering/LocalRewriter/LocalRewriter_PointerElementAccess.cs @@ -19,11 +19,6 @@ public override BoundNode VisitPointerElementAccess(BoundPointerElementAccess no } private BoundExpression RewritePointerElementAccess(BoundPointerElementAccess node, BoundExpression rewrittenExpression, BoundExpression rewrittenIndex) - { - return RewritePointerElementAccess(node, rewrittenExpression, rewrittenIndex, node.Checked); - } - - private BoundExpression RewritePointerElementAccess(BoundExpression node, BoundExpression rewrittenExpression, BoundExpression rewrittenIndex, bool @checked) { // Optimization: p[0] == *p if (rewrittenIndex.IsDefaultValue()) @@ -54,7 +49,7 @@ private BoundExpression RewritePointerElementAccess(BoundExpression node, BoundE throw ExceptionUtilities.UnexpectedValue(rewrittenIndex.Type.SpecialType); } - if (@checked) + if (node.Checked) { additionKind |= BinaryOperatorKind.Checked; } From 9ef47522bd7426f50a1c13c1e90197eeeb54fca3 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:31:16 +0330 Subject: [PATCH 08/43] Avoid binding the initializer twice --- .../CSharp/Portable/Binder/Binder_Expressions.cs | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index d521d0b94f661..bff15adade80f 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -2740,13 +2740,14 @@ private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitSt } return BindStackAllocWithInitializer( - node, - initializer, + node, + initializer, type: inLegalPosition ? GetStackAllocType(node, bestType, diagnostics) : null, elementType: bestType, - sizeOpt: null, + sizeOpt: null, diagnostics, - hasErrors: hasErrors); + hasErrors, + boundInitializerExpressions); } // This method binds all the array initializer expressions. @@ -3206,10 +3207,11 @@ private BoundExpression BindStackAllocWithInitializer( { if (boundInitExprOpt.IsDefault) { - boundInitExprOpt = BindArrayInitializerExpressions(initSyntax, diagnostics, dimension: 1, rank: 1) - .SelectAsArray((expr, t) => GenerateConversionForAssignment(t.elementType, expr, t.diagnostics), (elementType, diagnostics)); + boundInitExprOpt = BindArrayInitializerExpressions(initSyntax, diagnostics, dimension: 1, rank: 1); } + boundInitExprOpt = boundInitExprOpt.SelectAsArray((expr, t) => GenerateConversionForAssignment(t.elementType, expr, t.diagnostics), (elementType, diagnostics)); + if (sizeOpt != null) { int? constantSizeOpt = GetIntegerConstantForArraySize(sizeOpt); From 6e9010a475c8ac9579c06a3cb54b0218010488c4 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:31:46 +0330 Subject: [PATCH 09/43] Fix up formatting --- src/Compilers/CSharp/Portable/BoundTree/Formatting.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Portable/BoundTree/Formatting.cs b/src/Compilers/CSharp/Portable/BoundTree/Formatting.cs index 3bb5642f6a3e7..0f9f30c193bac 100644 --- a/src/Compilers/CSharp/Portable/BoundTree/Formatting.cs +++ b/src/Compilers/CSharp/Portable/BoundTree/Formatting.cs @@ -149,7 +149,7 @@ internal partial class BoundStackAllocArrayCreation { public override object Display { - get { return string.Format(MessageID.IDS_StackAllocExpression.Localize().ToString(), ElementType, Count.Syntax); } + get { return string.Format(MessageID.IDS_StackAllocExpression.Localize().ToString(), ElementType, Count.WasCompilerGenerated ? null : Count.Syntax); } } } } From 47a58fff811516fe0b4b520f7d22003f8989a5a6 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:32:44 +0330 Subject: [PATCH 10/43] Add flowpass tests --- .../Semantics/StackAllocInitializerTests.cs | 51 +++++++++++++++++++ 1 file changed, 51 insertions(+) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index c9d2bf7a0f6b0..e3c9f8f33faab 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -246,6 +246,57 @@ public void Method1() ); } + [Fact] + public void TestFlowPass1() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public static void Main() + { + int i, j, k; + var obj1 = stackalloc int [1] { i = 1 }; + var obj2 = stackalloc int [ ] { j = 2 }; + var obj3 = stackalloc [ ] { k = 3 }; + + Console.Write(i); + Console.Write(j); + Console.Write(k); + } +}", TestOptions.UnsafeReleaseExe); + + CompileAndVerify(comp, expectedOutput: "123"); + } + + [Fact] + public void TestFlowPass2() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + public static void Main() + { + int i, j, k; + var obj1 = stackalloc int [1] { i }; + var obj2 = stackalloc int [ ] { j }; + var obj3 = stackalloc [ ] { k }; + } +}", TestOptions.UnsafeReleaseExe); + + comp.VerifyDiagnostics( + // (7,41): error CS0165: Use of unassigned local variable 'i' + // var obj1 = stackalloc int [1] { i }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "i").WithArguments("i").WithLocation(7, 41), + // (8,41): error CS0165: Use of unassigned local variable 'j' + // var obj2 = stackalloc int [ ] { j }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "j").WithArguments("j").WithLocation(8, 41), + // (9,41): error CS0165: Use of unassigned local variable 'k' + // var obj3 = stackalloc [ ] { k }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "k").WithArguments("k").WithLocation(9, 41) + ); + } + [Fact] public void ConversionFromPointerStackAlloc_UserDefined_Implicit() { From d641f408df248f6a6887fa8a26ee2e93831f40b4 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 19:32:59 +0330 Subject: [PATCH 11/43] Clean up tests --- .../Semantics/StackAllocInitializerTests.cs | 392 ++++++++++-------- 1 file changed, 220 insertions(+), 172 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index e3c9f8f33faab..dae66f62a87b2 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -150,25 +150,22 @@ unsafe class Test { public void Method1() { - var obj1 = stackalloc[] {{1}}; - var obj2 = stackalloc int[] {{1}}; - var obj3 = stackalloc int[1] {{1}}; + var obj1 = stackalloc int[1] { { 42 } }; + var obj2 = stackalloc int[ ] { { 42 } }; + var obj3 = stackalloc [ ] { { 42 } }; } }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( - // (6,34): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. - // var obj1 = stackalloc[] {{1}}; - Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34), - // (6,34): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. - // var obj1 = stackalloc[] {{1}}; - Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(6, 34), - // (7,38): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. - // var obj2 = stackalloc int[] {{1}}; - Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(7, 38), - // (8,39): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. - // var obj3 = stackalloc int[1] {{1}}; - Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{1}").WithLocation(8, 39) + // (6,40): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj1 = stackalloc int[1] { { 42 } }; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{ 42 }").WithLocation(6, 40), + // (7,40): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj2 = stackalloc int[ ] { { 42 } }; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{ 42 }").WithLocation(7, 40), + // (8,40): error CS0623: Array initializers can only be used in a variable or field initializer. Try using a new expression instead. + // var obj3 = stackalloc [ ] { { 42 } }; + Diagnostic(ErrorCode.ERR_ArrayInitInBadPlace, "{ 42 }").WithLocation(8, 40) ); } @@ -523,12 +520,11 @@ void Method2() void Method3() { - double x = stackalloc[] { 1, 2, 3 }; // implicit - short y = (short)stackalloc[] { 1, 2, 3 }; // explicit + double x = stackalloc[] { 1, 2, 3 }; // implicit + short y = (short)stackalloc[] { 1, 2, 3 }; // explicit } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( - // (6,20): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double' is not possible. // double x = stackalloc int[3] { 1, 2, 3 }; // implicit Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "double").WithLocation(6, 20), @@ -558,29 +554,29 @@ class Test { void M() { - Span a1 = stackalloc int[3] { 1, 2, 3 }; - Span a2 = stackalloc int[] { 1, 2, 3 }; - Span a3 = stackalloc[] { 1, 2, 3 }; + Span a1 = stackalloc int [3] { 1, 2, 3 }; + Span a2 = stackalloc int [ ] { 1, 2, 3 }; + Span a3 = stackalloc [ ] { 1, 2, 3 }; } }").VerifyDiagnostics( // (6,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) - // Span a1 = stackalloc int[3] { 1, 2, 3 }; + // Span a1 = stackalloc int [3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(6, 9), // (6,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported - // Span a1 = stackalloc int[3] { 1, 2, 3 }; - Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int[3] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(6, 24), + // Span a1 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int [3] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(6, 24), // (7,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) - // Span a2 = stackalloc int[] { 1, 2, 3 }; + // Span a2 = stackalloc int [ ] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(7, 9), // (7,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported - // Span a2 = stackalloc int[] { 1, 2, 3 }; - Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int[] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(7, 24), + // Span a2 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc int [ ] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(7, 24), // (8,9): error CS0246: The type or namespace name 'Span<>' could not be found (are you missing a using directive or an assembly reference?) - // Span a3 = stackalloc[] { 1, 2, 3 }; + // Span a3 = stackalloc [ ] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "Span").WithArguments("Span<>").WithLocation(8, 9), // (8,24): error CS0518: Predefined type 'System.Span`1' is not defined or imported - // Span a3 = stackalloc[] { 1, 2, 3 }; - Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc[] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(8, 24) + // Span a3 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_PredefinedTypeNotFound, "stackalloc [ ] { 1, 2, 3 }").WithArguments("System.Span`1").WithLocation(8, 24) ); } @@ -598,8 +594,8 @@ class Test void M() { Span a1 = stackalloc int [3] { 1, 2, 3 }; - Span a2 = stackalloc int [] { 1, 2, 3 }; - Span a3 = stackalloc [] { 1, 2, 3 }; + Span a2 = stackalloc int [ ] { 1, 2, 3 }; + Span a3 = stackalloc [ ] { 1, 2, 3 }; } } }").VerifyDiagnostics( @@ -607,11 +603,11 @@ void M() // Span a1 = stackalloc int [3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [3] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(11, 28), // (12,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' - // Span a2 = stackalloc int [] { 1, 2, 3 }; - Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(12, 28), + // Span a2 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [ ] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(12, 28), // (13,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' - // Span a3 = stackalloc [] { 1, 2, 3 }; - Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc [] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(13, 28) + // Span a3 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc [ ] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(13, 28) ); } @@ -623,9 +619,9 @@ class Test { void M() { - var x1 = true ? stackalloc int [3] { 1, 2, 3, } : stackalloc int [3] { 1, 2, 3, }; - var x2 = true ? stackalloc int [] { 1, 2, 3, } : stackalloc int [] { 1, 2, 3, }; - var x3 = true ? stackalloc [] { 1, 2, 3, } : stackalloc [] { 1, 2, 3, }; + var x1 = true ? stackalloc int [3] { 1, 2, 3 } : stackalloc int [3] { 1, 2, 3 }; + var x2 = true ? stackalloc int [ ] { 1, 2, 3 } : stackalloc int [ ] { 1, 2, 3 }; + var x3 = true ? stackalloc [ ] { 1, 2, 3 } : stackalloc [ ] { 1, 2, 3 }; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); } @@ -639,9 +635,9 @@ class Test { void M() { - var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc int [3] { 1, 2, 3, }; - var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc int [] { 1, 2, 3, }; - var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { 1, 2, 3, }; + var x1 = true ? stackalloc int [3] { 1, 2, 3 } : (Span)stackalloc int [3] { 1, 2, 3 }; + var x2 = true ? stackalloc int [ ] { 1, 2, 3 } : (Span)stackalloc int [ ] { 1, 2, 3 }; + var x3 = true ? stackalloc [ ] { 1, 2, 3 } : (Span)stackalloc [ ] { 1, 2, 3 }; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); } @@ -656,19 +652,19 @@ class Test void M() { var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc short [3] { (short)1, (short)2, (short)3 }; - var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc short [] { (short)1, (short)2, (short)3 }; - var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { (short)1, (short)2, (short)3 }; + var x2 = true ? stackalloc int [ ] { 1, 2, 3, } : (Span)stackalloc short [ ] { (short)1, (short)2, (short)3 }; + var x3 = true ? stackalloc [ ] { 1, 2, 3, } : (Span)stackalloc [ ] { (short)1, (short)2, (short)3 }; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( // (7,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. // var x1 = true ? stackalloc int [3] { 1, 2, 3, } : (Span)stackalloc short [3] { (short)1, (short)2, (short)3 }; Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc short [3] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(7, 59), // (8,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. - // var x2 = true ? stackalloc int [] { 1, 2, 3, } : (Span)stackalloc short [] { (short)1, (short)2, (short)3 }; - Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc short [] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(8, 59), + // var x2 = true ? stackalloc int [ ] { 1, 2, 3, } : (Span)stackalloc short [ ] { (short)1, (short)2, (short)3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc short [ ] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(8, 59), // (9,59): error CS8346: Conversion of a stackalloc expression of type 'short' to type 'Span' is not possible. - // var x3 = true ? stackalloc [] { 1, 2, 3, } : (Span)stackalloc [] { (short)1, (short)2, (short)3 }; - Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc [] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(9, 59) + // var x3 = true ? stackalloc [ ] { 1, 2, 3, } : (Span)stackalloc [ ] { (short)1, (short)2, (short)3 }; + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "(Span)stackalloc [ ] { (short)1, (short)2, (short)3 }").WithArguments("short", "System.Span").WithLocation(9, 59) ); } @@ -681,8 +677,13 @@ class Test { void M() { - Span a = stackalloc int [10]; - var x = true ? stackalloc int [10] : a; + Span a1 = stackalloc int [3] { 1, 2, 3 }; + Span a2 = stackalloc int [ ] { 1, 2, 3 }; + Span a3 = stackalloc [ ] { 1, 2, 3 }; + + var x1 = true ? stackalloc int [3] { 1, 2, 3 } : a1; + var x2 = true ? stackalloc int [ ] { 1, 2, 3 } : a2; + var x3 = true ? stackalloc [ ] { 1, 2, 3 } : a3; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); } @@ -698,19 +699,19 @@ void M() { Span a = stackalloc short [10]; var x1 = true ? stackalloc int [3] { 1, 2, 3 } : a; - var x2 = true ? stackalloc int [] { 1, 2, 3 } : a; - var x3 = true ? stackalloc [] { 1, 2, 3 } : a; + var x2 = true ? stackalloc int [ ] { 1, 2, 3 } : a; + var x3 = true ? stackalloc [ ] { 1, 2, 3 } : a; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( // (8,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[3]' and 'Span' // var x1 = true ? stackalloc int [3] { 1, 2, 3 } : a; Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc int [3] { 1, 2, 3 } : a").WithArguments("stackalloc int[3]", "System.Span").WithLocation(8, 18), - // (9,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[stackalloc int [] { 1, 2, 3 }]' and 'Span' - // var x2 = true ? stackalloc int [] { 1, 2, 3 } : a; - Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc int [] { 1, 2, 3 } : a").WithArguments("stackalloc int[stackalloc int [] { 1, 2, 3 }]", "System.Span").WithLocation(9, 18), - // (10,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[stackalloc [] { 1, 2, 3 }]' and 'Span' - // var x3 = true ? stackalloc [] { 1, 2, 3 } : a; - Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc [] { 1, 2, 3 } : a").WithArguments("stackalloc int[stackalloc [] { 1, 2, 3 }]", "System.Span").WithLocation(10, 18) + // (9,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[]' and 'Span' + // var x2 = true ? stackalloc int [ ] { 1, 2, 3 } : a; + Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc int [ ] { 1, 2, 3 } : a").WithArguments("stackalloc int[]", "System.Span").WithLocation(9, 18), + // (10,18): error CS0173: Type of conditional expression cannot be determined because there is no implicit conversion between 'stackalloc int[]' and 'Span' + // var x3 = true ? stackalloc [ ] { 1, 2, 3 } : a; + Diagnostic(ErrorCode.ERR_InvalidQM, "true ? stackalloc [ ] { 1, 2, 3 } : a").WithArguments("stackalloc int[]", "System.Span").WithLocation(10, 18) ); } @@ -726,13 +727,13 @@ void M() { var x = N() ? N() - ? stackalloc int [1] { 2 } - : stackalloc int[] { 2 } + ? stackalloc int[1] { 42 } + : stackalloc int[ ] { 42 } : N() - ? stackalloc[] { 2 } + ? stackalloc[] { 42 } : N() - ? stackalloc int[4] - : stackalloc int[5]; + ? stackalloc int[2] + : stackalloc int[3]; } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); } @@ -745,29 +746,29 @@ class Test { void M() { - if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } - if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } - if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } + if (stackalloc int[3] { 1, 2, 3 } == stackalloc int[3] { 1, 2, 3 }) { } + if (stackalloc int[ ] { 1, 2, 3 } == stackalloc int[ ] { 1, 2, 3 }) { } + if (stackalloc [ ] { 1, 2, 3 } == stackalloc [ ] { 1, 2, 3 }) { } } }", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( - // (6,12): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 12), - // (6,39): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc int[1] { 1 } == stackalloc int[1] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 39), - // (7,12): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 12), - // (7,39): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc int[] { 1 } == stackalloc int[] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 39), - // (8,12): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 12), - // (8,39): error CS1525: Invalid expression term 'stackalloc' - // if(stackalloc [] { 1 } == stackalloc [] { 2 }) { } - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 39) + // (6,13): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc int[3] { 1, 2, 3 } == stackalloc int[3] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 13), + // (6,46): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc int[3] { 1, 2, 3 } == stackalloc int[3] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 46), + // (7,13): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc int[ ] { 1, 2, 3 } == stackalloc int[ ] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 13), + // (7,46): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc int[ ] { 1, 2, 3 } == stackalloc int[ ] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 46), + // (8,13): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc [ ] { 1, 2, 3 } == stackalloc [ ] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 13), + // (8,46): error CS1525: Invalid expression term 'stackalloc' + // if (stackalloc [ ] { 1, 2, 3 } == stackalloc [ ] { 1, 2, 3 }) { } + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 46) ); } @@ -782,9 +783,9 @@ class Test { void M() { - Span x1 = stackalloc int[1] { 2 }; - Span x2 = stackalloc int[] { 2 }; - Span x3 = stackalloc [] { 2 }; + Span x1 = stackalloc int [3] { 1, 2, 3 }; + Span x2 = stackalloc int [ ] { 1, 2, 3 }; + Span x3 = stackalloc [ ] { 1, 2, 3 }; } }", options: TestOptions.UnsafeReleaseDll, parseOptions: parseOptions).VerifyDiagnostics( // (7,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. @@ -807,20 +808,20 @@ class Test { void M() { - var x1 = stackalloc int[1] { 2 }; - var x2 = stackalloc int[] { 2 }; - var x3 = stackalloc [] { 2 }; + var x1 = stackalloc int [3] { 1, 2, 3 }; + var x2 = stackalloc int [ ] { 1, 2, 3 }; + var x3 = stackalloc [ ] { 1, 2, 3 }; } }", options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( // (6,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context - // var x1 = stackalloc int[1] { 2 }; - Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[1] { 2 }").WithLocation(6, 18), + // var x1 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int [3] { 1, 2, 3 }").WithLocation(6, 18), // (7,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context - // var x2 = stackalloc int[] { 2 }; - Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[] { 2 }").WithLocation(7, 18), + // var x2 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(7, 18), // (8,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context - // var x3 = stackalloc [] { 2 }; - Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc [] { 2 }").WithLocation(8, 18) + // var x3 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc [ ] { 1, 2, 3 }").WithLocation(8, 18) ); } @@ -832,9 +833,9 @@ public class Test { unsafe public static void Main() { - using (var v1 = stackalloc int[1] { 2 } ) - using (var v2 = stackalloc int[] { 2 } ) - using (var v3 = stackalloc [] { 2 } ) + using (var v1 = stackalloc int [3] { 1, 2, 3 }) + using (var v2 = stackalloc int [ ] { 1, 2, 3 }) + using (var v3 = stackalloc [ ] { 1, 2, 3 }) { } } @@ -842,14 +843,14 @@ unsafe public static void Main() "; CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( // (6,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' - // using (var v1 = stackalloc int[1] { 2 } ) - Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v1 = stackalloc int[1] { 2 }").WithArguments("int*").WithLocation(6, 16), + // using (var v1 = stackalloc int [3] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v1 = stackalloc int [3] { 1, 2, 3 }").WithArguments("int*").WithLocation(6, 16), // (7,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' - // using (var v2 = stackalloc int[] { 2 } ) - Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v2 = stackalloc int[] { 2 }").WithArguments("int*").WithLocation(7, 16), + // using (var v2 = stackalloc int [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v2 = stackalloc int [ ] { 1, 2, 3 }").WithArguments("int*").WithLocation(7, 16), // (8,16): error CS1674: 'int*': type used in a using statement must be implicitly convertible to 'System.IDisposable' - // using (var v3 = stackalloc [] { 2 } ) - Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v3 = stackalloc [] { 2 }").WithArguments("int*").WithLocation(8, 16) + // using (var v3 = stackalloc [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_NoConvToIDisp, "var v3 = stackalloc [ ] { 1, 2, 3 }").WithArguments("int*").WithLocation(8, 16) ); } @@ -861,9 +862,9 @@ public class Test { unsafe public static void Main() { - using (System.IDisposable v1 = stackalloc int[1] { 2 } ) - using (System.IDisposable v2 = stackalloc int[] { 2 } ) - using (System.IDisposable v3 = stackalloc [] { 2 } ) + using (System.IDisposable v1 = stackalloc int [3] { 1, 2, 3 }) + using (System.IDisposable v2 = stackalloc int [ ] { 1, 2, 3 }) + using (System.IDisposable v3 = stackalloc [ ] { 1, 2, 3 }) { } } @@ -871,14 +872,43 @@ unsafe public static void Main() "; CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( // (6,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. - // using (System.IDisposable v1 = stackalloc int[1] { 2 } ) - Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[1] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(6, 40), + // using (System.IDisposable v1 = stackalloc int [3] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int [3] { 1, 2, 3 }").WithArguments("int", "System.IDisposable").WithLocation(6, 40), // (7,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. - // using (System.IDisposable v2 = stackalloc int[] { 2 } ) - Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(7, 40), + // using (System.IDisposable v2 = stackalloc int [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int [ ] { 1, 2, 3 }").WithArguments("int", "System.IDisposable").WithLocation(7, 40), // (8,40): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'IDisposable' is not possible. - // using (System.IDisposable v3 = stackalloc [] { 2 } ) - Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc [] { 2 }").WithArguments("int", "System.IDisposable").WithLocation(8, 40) + // using (System.IDisposable v3 = stackalloc [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc [ ] { 1, 2, 3 }").WithArguments("int", "System.IDisposable").WithLocation(8, 40) + ); + } + + [Fact] + public void StackAllocInFixed() + { + var test = @" +public class Test +{ + unsafe public static void Main() + { + fixed (int* v1 = stackalloc int [3] { 1, 2, 3 }) + fixed (int* v2 = stackalloc int [ ] { 1, 2, 3 }) + fixed (int* v3 = stackalloc [ ] { 1, 2, 3 }) + { + } + } +} +"; + CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( + // (6,26): error CS0213: You cannot use the fixed statement to take the address of an already fixed expression + // fixed (int* v1 = stackalloc int [3] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_FixedNotNeeded, "stackalloc int [3] { 1, 2, 3 }").WithLocation(6, 26), + // (7,26): error CS0213: You cannot use the fixed statement to take the address of an already fixed expression + // fixed (int* v2 = stackalloc int [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_FixedNotNeeded, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(7, 26), + // (8,26): error CS0213: You cannot use the fixed statement to take the address of an already fixed expression + // fixed (int* v3 = stackalloc [ ] { 1, 2, 3 }) + Diagnostic(ErrorCode.ERR_FixedNotNeeded, "stackalloc [ ] { 1, 2, 3 }").WithLocation(8, 26) ); } @@ -890,9 +920,9 @@ unsafe public class Test { void M() { - const int* p1 = stackalloc int[1] { 1 }; - const int* p2 = stackalloc int[] { 1 }; - const int* p3 = stackalloc [] { 1 }; + const int* p1 = stackalloc int [3] { 1, 2, 3 }; + const int* p2 = stackalloc int [ ] { 1, 2, 3 }; + const int* p3 = stackalloc [ ] { 1, 2, 3 }; } } "; @@ -918,31 +948,31 @@ public class Test { void M() { - ref Span p1 = stackalloc int[1] { 1 }; - ref Span p2 = stackalloc int[] { 1 }; - ref Span p3 = stackalloc [] { 1 }; + ref Span p1 = stackalloc int [3] { 1, 2, 3 }; + ref Span p2 = stackalloc int [ ] { 1, 2, 3 }; + ref Span p3 = stackalloc [ ] { 1, 2, 3 }; } } "; CreateCompilationWithMscorlibAndSpan(test, options: TestOptions.ReleaseDll.WithAllowUnsafe(true)).VerifyDiagnostics( // (7,23): error CS8172: Cannot initialize a by-reference variable with a value - // ref Span p1 = stackalloc int[1] { 1 }; - Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p1 = stackalloc int[1] { 1 }").WithLocation(7, 23), + // ref Span p1 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p1 = stackalloc int [3] { 1, 2, 3 }").WithLocation(7, 23), // (7,28): error CS1510: A ref or out value must be an assignable variable - // ref Span p1 = stackalloc int[1] { 1 }; - Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int[1] { 1 }").WithLocation(7, 28), + // ref Span p1 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int [3] { 1, 2, 3 }").WithLocation(7, 28), // (8,23): error CS8172: Cannot initialize a by-reference variable with a value - // ref Span p2 = stackalloc int[] { 1 }; - Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p2 = stackalloc int[] { 1 }").WithLocation(8, 23), + // ref Span p2 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p2 = stackalloc int [ ] { 1, 2, 3 }").WithLocation(8, 23), // (8,28): error CS1510: A ref or out value must be an assignable variable - // ref Span p2 = stackalloc int[] { 1 }; - Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int[] { 1 }").WithLocation(8, 28), + // ref Span p2 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(8, 28), // (9,23): error CS8172: Cannot initialize a by-reference variable with a value - // ref Span p3 = stackalloc [] { 1 }; - Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p3 = stackalloc [] { 1 }").WithLocation(9, 23), + // ref Span p3 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InitializeByReferenceVariableWithValue, "p3 = stackalloc [ ] { 1, 2, 3 }").WithLocation(9, 23), // (9,28): error CS1510: A ref or out value must be an assignable variable - // ref Span p3 = stackalloc [] { 1 }; - Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc [] { 1 }").WithLocation(9, 28) + // ref Span p3 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_RefLvalueExpected, "stackalloc [ ] { 1, 2, 3 }").WithLocation(9, 28) ); } @@ -956,8 +986,8 @@ public class Test void M() { ref Span p1 = ref stackalloc int [3] { 1, 2, 3 }; - ref Span p2 = ref stackalloc int [] { 1, 2, 3 }; - ref Span p3 = ref stackalloc [] { 1, 2, 3 }; + ref Span p2 = ref stackalloc int [ ] { 1, 2, 3 }; + ref Span p3 = ref stackalloc [ ] { 1, 2, 3 }; } } "; @@ -984,8 +1014,8 @@ public class Test void M() { N(stackalloc int [3] { 1, 2, 3 }); - N(stackalloc int [] { 1, 2, 3 }); - N(stackalloc [] { 1, 2, 3 }); + N(stackalloc int [ ] { 1, 2, 3 }); + N(stackalloc [ ] { 1, 2, 3 }); } void N(Span span) { @@ -1014,8 +1044,12 @@ public class Test void M() { int length1 = (stackalloc int [3] { 1, 2, 3 }).Length; - int length2 = (stackalloc int [] { 1, 2, 3 }).Length; - int length3 = (stackalloc [] { 1, 2, 3 }).Length; + int length2 = (stackalloc int [ ] { 1, 2, 3 }).Length; + int length3 = (stackalloc [ ] { 1, 2, 3 }).Length; + + int length4 = stackalloc int [3] { 1, 2, 3 }.Length; + int length5 = stackalloc int [ ] { 1, 2, 3 }.Length; + int length6 = stackalloc [ ] { 1, 2, 3 }.Length; } } "; @@ -1024,11 +1058,20 @@ void M() // int length1 = (stackalloc int [3] { 1, 2, 3 }).Length; Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 24), // (7,24): error CS1525: Invalid expression term 'stackalloc' - // int length2 = (stackalloc int [] { 1, 2, 3 }).Length; + // int length2 = (stackalloc int [ ] { 1, 2, 3 }).Length; Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(7, 24), // (8,24): error CS1525: Invalid expression term 'stackalloc' - // int length3 = (stackalloc [] { 1, 2, 3 }).Length; - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 24) + // int length3 = (stackalloc [ ] { 1, 2, 3 }).Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(8, 24), + // (10,23): error CS1525: Invalid expression term 'stackalloc' + // int length4 = stackalloc int [3] { 1, 2, 3 }.Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(10, 23), + // (11,23): error CS1525: Invalid expression term 'stackalloc' + // int length5 = stackalloc int [ ] { 1, 2, 3 }.Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(11, 23), + // (12,23): error CS1525: Invalid expression term 'stackalloc' + // int length6 = stackalloc [ ] { 1, 2, 3 }.Length; + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(12, 23) ); } @@ -1041,9 +1084,9 @@ unsafe public class Test { static void Main() { - Invoke(stackalloc int[3] { 1, 2, 3 }); - Invoke(stackalloc int[] { 1, 2, 3 }); - Invoke(stackalloc[] { 1, 2, 3 }); + Invoke(stackalloc int [3] { 1, 2, 3 }); + Invoke(stackalloc int [ ] { 1, 2, 3 }); + Invoke(stackalloc [ ] { 1, 2, 3 }); } static void Invoke(Span shortSpan) => Console.WriteLine(""shortSpan""); @@ -1074,23 +1117,26 @@ class Program static void Main() { dynamic d = 1; - var d1 = stackalloc dynamic[1] { d }; - var d2 = stackalloc dynamic[] { d }; - var d3 = stackalloc[] { d }; + var d1 = stackalloc dynamic [3] { d }; + var d2 = stackalloc dynamic [ ] { d }; + var d3 = stackalloc [ ] { d }; } }").VerifyDiagnostics( // (7,29): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // var d1 = stackalloc dynamic[1] { d }; + // var d1 = stackalloc dynamic [3] { d }; Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(7, 29), + // (7,18): error CS0847: An array initializer of length '3' is expected + // var d1 = stackalloc dynamic [3] { d }; + Diagnostic(ErrorCode.ERR_ArrayInitializerIncorrectLength, "stackalloc dynamic [3] { d }").WithArguments("3").WithLocation(7, 18), // (8,29): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // var d2 = stackalloc dynamic[] { d }; + // var d2 = stackalloc dynamic [ ] { d }; Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(8, 29), // (9,18): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // var d3 = stackalloc[] { d }; - Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(9, 18), + // var d3 = stackalloc [ ] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc [ ] { d }").WithArguments("dynamic").WithLocation(9, 18), // (9,18): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context - // var d3 = stackalloc[] { d }; - Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc[] { d }").WithLocation(9, 18) + // var d3 = stackalloc [ ] { d }; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc [ ] { d }").WithLocation(9, 18) ); } @@ -1104,20 +1150,23 @@ class Program static void Main() { dynamic d = 1; - Span d1 = stackalloc dynamic[1] { d }; - Span d2 = stackalloc dynamic[] { d }; - Span d3 = stackalloc[] { d }; + Span d1 = stackalloc dynamic [3] { d }; + Span d2 = stackalloc dynamic [ ] { d }; + Span d3 = stackalloc [ ] { d }; } }").VerifyDiagnostics( // (8,39): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // Span d1 = stackalloc dynamic[1] { d }; + // Span d1 = stackalloc dynamic [3] { d }; Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(8, 39), + // (8,28): error CS0847: An array initializer of length '3' is expected + // Span d1 = stackalloc dynamic [3] { d }; + Diagnostic(ErrorCode.ERR_ArrayInitializerIncorrectLength, "stackalloc dynamic [3] { d }").WithArguments("3").WithLocation(8, 28), // (9,39): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // Span d2 = stackalloc dynamic[] { d }; + // Span d2 = stackalloc dynamic [ ] { d }; Diagnostic(ErrorCode.ERR_ManagedAddr, "dynamic").WithArguments("dynamic").WithLocation(9, 39), // (10,28): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') - // Span d3 = stackalloc[] { d }; - Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(10, 28) + // Span d3 = stackalloc [ ] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc [ ] { d }").WithArguments("dynamic").WithLocation(10, 28) ); } @@ -1131,9 +1180,9 @@ static void N(object p) { } static void Main() { - N(stackalloc int[3] { 1, 2, 3 }); - N(stackalloc int[] { 1, 2, 3 }); - N(stackalloc[] { 1, 2, 3 }); + N(stackalloc int [3] { 1, 2, 3 }); + N(stackalloc int [ ] { 1, 2, 3 }); + N(stackalloc [ ] { 1, 2, 3 }); } }").VerifyDiagnostics( // (8,11): error CS1525: Invalid expression term 'stackalloc' @@ -1156,9 +1205,9 @@ class Program { static void Main() { - var x1 = (stackalloc int[3] { 1, 2, 3 }); - var x2 = (stackalloc int[] { 1, 2, 3 }); - var x3 = (stackalloc[] { 1, 2, 3 }); + var x1 = (stackalloc int [3] { 1, 2, 3 }); + var x2 = (stackalloc int [ ] { 1, 2, 3 }); + var x3 = (stackalloc [ ] { 1, 2, 3 }); } }").VerifyDiagnostics( // (6,19): error CS1525: Invalid expression term 'stackalloc' @@ -1182,8 +1231,8 @@ class Program static void Main() { var x1 = stackalloc int [3] { 1, 2, 3 } ?? stackalloc int [3] { 1, 2, 3 }; - var x2 = stackalloc int [] { 1, 2, 3 } ?? stackalloc int [] { 1, 2, 3 }; - var x3 = stackalloc [] { 1, 2, 3 } ?? stackalloc [] { 1, 2, 3 }; + var x2 = stackalloc int [ ] { 1, 2, 3 } ?? stackalloc int [ ] { 1, 2, 3 }; + var x3 = stackalloc [ ] { 1, 2, 3 } ?? stackalloc [ ] { 1, 2, 3 }; } }").VerifyDiagnostics( // (6,18): error CS1525: Invalid expression term 'stackalloc' @@ -1216,9 +1265,9 @@ class Test { public void Method() { - Test value1 = true ? new Test() : (Test)stackalloc int[3] { 1, 2, 3 }; - Test value2 = true ? new Test() : (Test)stackalloc int[] { 1, 2, 3 }; - Test value3 = true ? new Test() : (Test)stackalloc[] { 1, 2, 3 }; + Test value1 = true ? new Test() : (Test)stackalloc int [3] { 1, 2, 3 }; + Test value2 = true ? new Test() : (Test)stackalloc int [ ] { 1, 2, 3 }; + Test value3 = true ? new Test() : (Test)stackalloc [ ] { 1, 2, 3 }; } public static explicit operator Test(Span value) @@ -1228,5 +1277,4 @@ public static explicit operator Test(Span value) }", TestOptions.ReleaseDll).VerifyDiagnostics(); } } - } From aa4b04d06975c7313a9b6f06d03e7e6bd65c219a Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 1 Feb 2018 21:08:38 +0330 Subject: [PATCH 12/43] Fix typo --- src/Compilers/CSharp/Portable/CSharpResources.Designer.cs | 2 +- src/Compilers/CSharp/Portable/CSharpResources.resx | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf | 4 ++-- src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf | 4 ++-- .../CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf | 4 ++-- .../CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf | 4 ++-- .../Test/Semantic/Semantics/StackAllocInitializerTests.cs | 6 +++--- .../Test/Syntax/Parsing/StackAllocInitializerTests.cs | 6 +++--- 17 files changed, 35 insertions(+), 35 deletions(-) diff --git a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs index 8bceed68e9662..a0980827131e8 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs +++ b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs @@ -10746,7 +10746,7 @@ internal static string IDS_FeatureRefStructs { } /// - /// Looks up a localized string similar to stackalloc initilizer. + /// Looks up a localized string similar to stackalloc initializer. /// internal static string IDS_FeatureStackAllocInitializer { get { diff --git a/src/Compilers/CSharp/Portable/CSharpResources.resx b/src/Compilers/CSharp/Portable/CSharpResources.resx index be6f120074e14..e2e8c61b0d2ad 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.resx +++ b/src/Compilers/CSharp/Portable/CSharpResources.resx @@ -5254,6 +5254,6 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Arguments with 'in' modifier cannot be used in dynamically dispatched expessions. - stackalloc initilizer + stackalloc initializer - + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf index 9f9aa698dd242..b589548dce920 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf @@ -8611,8 +8611,8 @@ Pokud chcete odstranit toto varování, můžete místo toho použít /reference - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf index 8552a32a89502..1dcb27e2caed8 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf @@ -8611,8 +8611,8 @@ Um die Warnung zu beheben, können Sie stattdessen /reference verwenden (Einbett - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf index 6c7b8cb95e918..5765b6ad8490a 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf @@ -8611,8 +8611,8 @@ Para eliminar la advertencia puede usar /reference (establezca la propiedad Embe - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf index 4f2227605ac2d..dad97f8fed2f7 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf @@ -8611,8 +8611,8 @@ Pour supprimer l'avertissement, vous pouvez utiliser la commande /reference (dé - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf index 9145240bd7dfd..0448ebf73ddf4 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf @@ -8611,8 +8611,8 @@ Per rimuovere l'avviso, è invece possibile usare /reference (impostare la propr - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf index 11c69dadec18f..904a3e6402123 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf @@ -8611,8 +8611,8 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf index 45b0457798ae0..873df201acb85 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf @@ -8611,8 +8611,8 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf index c80d8402301ee..01f7c3621a3e7 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf @@ -8611,8 +8611,8 @@ Aby usunąć ostrzeżenie, możesz zamiast tego użyć opcji /reference (ustaw w - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf index 7ad7af67c55a1..7dd155d15d83d 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf @@ -8611,8 +8611,8 @@ Para incorporar informações de tipo de interoperabilidade para os dois assembl - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf index b7de0efd1d989..8132882ecad60 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf @@ -8611,8 +8611,8 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf index 28d9774b9e13b..68bd6dd346d8f 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf @@ -8611,8 +8611,8 @@ Uyarıyı kaldırmak için, /reference kullanabilirsiniz (Birlikte Çalışma T - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf index e2b2f8762aeaf..a7a3e6b52c4b5 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf @@ -8611,8 +8611,8 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf index b434983a3054b..5d48d59a502e0 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf @@ -8611,8 +8611,8 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ - stackalloc initilizer - stackalloc initilizer + stackalloc initializer + stackalloc initializer diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index dae66f62a87b2..4e13f4d547e41 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -790,13 +790,13 @@ void M() }", options: TestOptions.UnsafeReleaseDll, parseOptions: parseOptions).VerifyDiagnostics( // (7,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. // Span x1 = stackalloc int[1] { 2 }; - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(7, 24), + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(7, 24), // (8,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. // Span x2 = stackalloc int[] { 2 }; - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(8, 24), + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(8, 24), // (9,24): error CS8107: Feature 'stackalloc initilizer' is not available in C# 7.0. Please use language version 7.3 or greater. // Span x3 = stackalloc [] { 2 }; - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(9, 24) + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(9, 24) ); } diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs index b6a9d915153ab..91c6425da3604 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs @@ -16,7 +16,7 @@ public StackAllocInitializerTests(ITestOutputHelper output) : base(output) { } public void StackAllocInitializer_01() { UsingExpression("stackalloc int[] { 42 }", options: TestOptions.Regular7, - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1)); N(SyntaxKind.StackAllocArrayCreationExpression); { N(SyntaxKind.StackAllocKeyword); @@ -53,7 +53,7 @@ public void StackAllocInitializer_01() public void StackAllocInitializer_02() { UsingExpression("stackalloc int[1] { 42 }", options: TestOptions.Regular7, - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1)); N(SyntaxKind.StackAllocArrayCreationExpression); { N(SyntaxKind.StackAllocKeyword); @@ -90,7 +90,7 @@ public void StackAllocInitializer_02() public void StackAllocInitializer_03() { UsingExpression("stackalloc[] { 42 }", options: TestOptions.Regular7, - Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initilizer", "7.3").WithLocation(1, 1)); + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1)); N(SyntaxKind.ImplicitStackAllocArrayCreationExpression); { N(SyntaxKind.StackAllocKeyword); From 7ff7236aa0c0f7530287f62780fc876d4579de84 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 08:54:30 +0330 Subject: [PATCH 13/43] Revert code. --- .../CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs index 3cace8bc0a95b..0615e1f626f34 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/ExpressionParsingTests.cs @@ -3,7 +3,6 @@ using System.Linq; using Microsoft.CodeAnalysis.CSharp.Syntax; using Microsoft.CodeAnalysis.CSharp.Test.Utilities; -using Microsoft.CodeAnalysis.Test.Utilities; using Roslyn.Test.Utilities; using Xunit; using Xunit.Abstractions; From facb834dcc6fad7bcfbcd515ec35e4746a435f49 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 09:11:32 +0330 Subject: [PATCH 14/43] Add parsing test for bad implicit stackalloc --- .../Parsing/StackAllocInitializerTests.cs | 50 +++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs index 91c6425da3604..e586f8a56e03f 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs @@ -108,5 +108,55 @@ public void StackAllocInitializer_03() } EOF(); } + + [Fact] + public void StackAllocInitializer_04() + { + UsingExpression("stackalloc[1] { 42 }", options: TestOptions.Regular7, + // (1,1): error CS8107: Feature 'stackalloc initializer' is not available in C# 7.0. Please use language version 7.3 or greater. + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1), + // (1,12): error CS1003: Syntax error, ']' expected + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_SyntaxError, "1").WithArguments("]", "").WithLocation(1, 12), + // (1,12): error CS1514: { expected + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_LbraceExpected, "1").WithLocation(1, 12), + // (1,13): error CS1003: Syntax error, ',' expected + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_SyntaxError, "]").WithArguments(",", "]").WithLocation(1, 13), + // (1,15): error CS1003: Syntax error, ',' expected + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments(",", "{").WithLocation(1, 15), + // (1,21): error CS1513: } expected + // stackalloc[1] { 42 } + Diagnostic(ErrorCode.ERR_RbraceExpected, "").WithLocation(1, 21)); + N(SyntaxKind.ImplicitStackAllocArrayCreationExpression); + { + N(SyntaxKind.StackAllocKeyword); + N(SyntaxKind.OpenBracketToken); + M(SyntaxKind.CloseBracketToken); + N(SyntaxKind.ArrayInitializerExpression); + { + M(SyntaxKind.OpenBraceToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "1"); + } + M(SyntaxKind.CommaToken); + N(SyntaxKind.ArrayInitializerExpression); + { + N(SyntaxKind.OpenBraceToken); + N(SyntaxKind.NumericLiteralExpression); + { + N(SyntaxKind.NumericLiteralToken, "42"); + } + N(SyntaxKind.CloseBraceToken); + } + M(SyntaxKind.CloseBraceToken); + } + } + EOF(); + } } } From f95fb81c903da3fdbc975dd5b7c41cfe29b927a3 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 09:12:14 +0330 Subject: [PATCH 15/43] Test stackalloc in catch and finally --- .../Portable/Binder/Binder_Expressions.cs | 12 +- .../Semantics/StackAllocInitializerTests.cs | 328 ++++++++++++++++++ 2 files changed, 334 insertions(+), 6 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index bff15adade80f..267b6bf207fd2 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -3071,12 +3071,6 @@ private BoundExpression BindStackAllocArrayCreationExpression( bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); bool hasErrors = !inLegalPosition; - // Check if we're syntactically within a catch or finally clause. - if (this.Flags.IncludesAny(BinderFlags.InCatchBlock | BinderFlags.InCatchFilter | BinderFlags.InFinallyBlock)) - { - Error(diagnostics, ErrorCode.ERR_StackallocInCatchFinally, node); - } - TypeSyntax typeSyntax = node.Type; if (typeSyntax.Kind() != SyntaxKind.ArrayType) @@ -3175,6 +3169,12 @@ private bool ReportBadStackAllocPosition(SyntaxNode node, DiagnosticBag diagnost SyntaxFacts.GetText(SyntaxKind.StackAllocKeyword)); } + // Check if we're syntactically within a catch or finally clause. + if (this.Flags.IncludesAny(BinderFlags.InCatchBlock | BinderFlags.InCatchFilter | BinderFlags.InFinallyBlock)) + { + Error(diagnostics, ErrorCode.ERR_StackallocInCatchFinally, node); + } + return inLegalPosition; } diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 4e13f4d547e41..d7ae4e07d2380 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -1276,5 +1276,333 @@ public static explicit operator Test(Span value) } }", TestOptions.ReleaseDll).VerifyDiagnostics(); } + + [Fact] + public void ERR_StackallocInCatchFinally_Catch() + { + var text = @" +unsafe class C +{ + int x = M(() => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* q1 = stackalloc int [3] { 1, 2, 3 }; + int* q2 = stackalloc int [ ] { 1, 2, 3 }; + int* q3 = stackalloc [ ] { 1, 2, 3 }; + } + catch + { + int* err11 = stackalloc int [3] { 1, 2, 3 }; + int* err12 = stackalloc int [ ] { 1, 2, 3 }; + int* err13 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + catch + { + int* err21 = stackalloc int [3] { 1, 2, 3 }; + int* err22 = stackalloc int [ ] { 1, 2, 3 }; + int* err23 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + } + catch + { + int* err31 = stackalloc int [3] { 1, 2, 3 }; + int* err32 = stackalloc int [ ] { 1, 2, 3 }; + int* err33 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + }); + + static int M(System.Action action) + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* q1 = stackalloc int [3] { 1, 2, 3 }; + int* q2 = stackalloc int [ ] { 1, 2, 3 }; + int* q3 = stackalloc [ ] { 1, 2, 3 }; + } + catch + { + int* err41 = stackalloc int [3] { 1, 2, 3 }; + int* err42 = stackalloc int [ ] { 1, 2, 3 }; + int* err43 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + catch + { + int* err51 = stackalloc int [3] { 1, 2, 3 }; + int* err52 = stackalloc int [ ] { 1, 2, 3 }; + int* err53 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + } + catch + { + int* err61 = stackalloc int [3] { 1, 2, 3 }; + int* err62 = stackalloc int [ ] { 1, 2, 3 }; + int* err63 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + return 0; + } +} +"; + CreateStandardCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (23,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err11 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(23, 34), + // (24,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err12 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(24, 34), + // (25,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err13 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(25, 34), + // (31,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err21 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(31, 26), + // (32,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err22 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(32, 26), + // (33,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err23 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(33, 26), + // (45,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err31 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(45, 34), + // (46,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err32 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(46, 34), + // (47,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err33 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(47, 34), + // (72,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err41 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(72, 34), + // (73,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err42 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(73, 34), + // (74,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err43 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(74, 34), + // (80,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err51 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(80, 26), + // (81,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err52 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(81, 26), + // (82,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err53 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(82, 26), + // (94,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err61 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(94, 34), + // (95,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err62 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(95, 34), + // (96,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err63 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(96, 34) + ); + } + + [Fact] + public void ERR_StackallocInCatchFinally_Finally() + { + var text = @" +unsafe class C +{ + int x = M(() => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* q1 = stackalloc int [3] { 1, 2, 3 }; + int* q2 = stackalloc int [ ] { 1, 2, 3 }; + int* q3 = stackalloc [ ] { 1, 2, 3 }; + } + finally + { + int* err11 = stackalloc int [3] { 1, 2, 3 }; + int* err12 = stackalloc int [ ] { 1, 2, 3 }; + int* err13 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + finally + { + int* err21 = stackalloc int [3] { 1, 2, 3 }; + int* err22 = stackalloc int [ ] { 1, 2, 3 }; + int* err23 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + } + finally + { + int* err31 = stackalloc int [3] { 1, 2, 3 }; + int* err32 = stackalloc int [ ] { 1, 2, 3 }; + int* err33 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + }); + + static int M(System.Action action) + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* q1 = stackalloc int [3] { 1, 2, 3 }; + int* q2 = stackalloc int [ ] { 1, 2, 3 }; + int* q3 = stackalloc [ ] { 1, 2, 3 }; + } + finally + { + int* err41 = stackalloc int [3] { 1, 2, 3 }; + int* err42 = stackalloc int [ ] { 1, 2, 3 }; + int* err43 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + finally + { + int* err51 = stackalloc int [3] { 1, 2, 3 }; + int* err52 = stackalloc int [ ] { 1, 2, 3 }; + int* err53 = stackalloc [ ] { 1, 2, 3 }; + System.Action a = () => + { + try + { + // fine + int* p1 = stackalloc int [3] { 1, 2, 3 }; + int* p2 = stackalloc int [ ] { 1, 2, 3 }; + int* p3 = stackalloc [ ] { 1, 2, 3 }; + } + finally + { + int* err61 = stackalloc int [3] { 1, 2, 3 }; + int* err62 = stackalloc int [ ] { 1, 2, 3 }; + int* err63 = stackalloc [ ] { 1, 2, 3 }; + } + }; + } + return 0; + } +} +"; + CreateStandardCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (23,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err11 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(23, 34), + // (24,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err12 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(24, 34), + // (25,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err13 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(25, 34), + // (31,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err21 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(31, 26), + // (32,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err22 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(32, 26), + // (33,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err23 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(33, 26), + // (45,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err31 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(45, 34), + // (46,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err32 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(46, 34), + // (47,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err33 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(47, 34), + // (72,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err41 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(72, 34), + // (73,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err42 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(73, 34), + // (74,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err43 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(74, 34), + // (80,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err51 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(80, 26), + // (81,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err52 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(81, 26), + // (82,26): error CS0255: stackalloc may not be used in a catch or finally block + // int* err53 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(82, 26), + // (94,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err61 = stackalloc int [3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(94, 34), + // (95,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err62 = stackalloc int [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [ ] { 1, 2, 3 }").WithLocation(95, 34), + // (96,34): error CS0255: stackalloc may not be used in a catch or finally block + // int* err63 = stackalloc [ ] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(96, 34) + ); + } } } From 6584454806d5d9a2a4e1cb63d6cef1c71750de5c Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 09:55:55 +0330 Subject: [PATCH 16/43] Whitespaces. --- src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs | 1 + .../CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs | 1 - 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index 267b6bf207fd2..f4cb3d8b7e356 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -3234,6 +3234,7 @@ private BoundExpression BindStackAllocWithInitializer( GetSpecialType(SpecialType.System_Int32, diagnostics, node)) { WasCompilerGenerated = true }; } + return new BoundStackAllocArrayCreation(node, elementType, sizeOpt, new BoundArrayInitialization(initSyntax, boundInitExprOpt), type, hasErrors); } diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index d7ae4e07d2380..545c848d72366 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -335,7 +335,6 @@ public static implicit operator Test(int* value) }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( - // (11,24): error CS8346: Conversion of a stackalloc expression of type 'int' to type 'double*' is not possible. // double* obj5 = stackalloc int[3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_StackAllocConversionNotPossible, "stackalloc int[3] { 1, 2, 3 }").WithArguments("int", "double*").WithLocation(11, 24), From 974ee8bc430232aac4ac969a90738c853c5dc26e Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 10:07:15 +0330 Subject: [PATCH 17/43] Add more best type tests --- .../Semantics/StackAllocInitializerTests.cs | 68 ++++++++++++------- 1 file changed, 42 insertions(+), 26 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 545c848d72366..9c7fcfa03f559 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -25,16 +25,47 @@ unsafe class Test struct A {} struct B {} - public void Method1() + public void Method() { - var obj1 = stackalloc[] { new A(), new B() }; + var p1 = stackalloc[] { new A(), new B() }; + var p2 = stackalloc[] { }; + var p3 = stackalloc[] { Method() }; + var p4 = stackalloc[] { null }; + var p5 = stackalloc[] { (1, null) }; + var p6 = stackalloc[] { () => { } }; + var p7 = stackalloc[] { new {} , new { i = 0 } }; + } -}", TestOptions.UnsafeReleaseDll); +} +namespace System { + public struct ValueTuple { + public ValueTuple(T1 a, T2 b) { } + } +} +", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( - // (9,20): error CS0826: No best type found for implicitly-typed array - // var obj1 = stackalloc[] { new A(), new B() }; - Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new A(), new B() }").WithLocation(9, 20) + // (9,18): error CS0826: No best type found for implicitly-typed array + // var p1 = stackalloc[] { new A(), new B() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new A(), new B() }").WithLocation(9, 18), + // (10,18): error CS0826: No best type found for implicitly-typed array + // var p2 = stackalloc[] { }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { }").WithLocation(10, 18), + // (11,18): error CS0826: No best type found for implicitly-typed array + // var p3 = stackalloc[] { Method() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { Method() }").WithLocation(11, 18), + // (12,18): error CS0826: No best type found for implicitly-typed array + // var p4 = stackalloc[] { null }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { null }").WithLocation(12, 18), + // (13,18): error CS0826: No best type found for implicitly-typed array + // var p5 = stackalloc[] { (1, null) }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { (1, null) }").WithLocation(13, 18), + // (14,18): error CS0826: No best type found for implicitly-typed array + // var p6 = stackalloc[] { () => { } }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { () => { } }").WithLocation(14, 18), + // (15,18): error CS0826: No best type found for implicitly-typed array + // var p7 = stackalloc[] { new {} , new { i = 0 } }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new {} , new { i = 0 } }").WithLocation(15, 18) ); } @@ -75,13 +106,17 @@ unsafe class Test public void Method1() { var obj1 = stackalloc[] { """" }; + var obj2 = stackalloc[] { new {} }; } }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( // (6,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('string') // var obj1 = stackalloc[] { "" }; - Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(6, 20) + Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(6, 20), + // (7,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('') + // var obj2 = stackalloc[] { new {} }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(7, 20) ); } @@ -123,25 +158,6 @@ public void Method1() ); } - [Fact] - public void NoBestType2() - { - var comp = CreateCompilationWithMscorlibAndSpan(@" -unsafe class Test -{ - public void Method1() - { - var obj1 = stackalloc[]{}; - } -}", TestOptions.UnsafeReleaseDll); - - comp.VerifyDiagnostics( - // (6,20): error CS0826: No best type found for implicitly-typed array - // var obj1 = stackalloc[]{}; - Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[]{}").WithLocation(6, 20) - ); - } - [Fact] public void NestedInit() { From dde9175cbc52f25e8c9be47cf94e95052d1fac69 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 10:12:11 +0330 Subject: [PATCH 18/43] Verify execution order --- .../CodeGen/StackAllocInitializerTests.cs | 26 ++++++++++++------- 1 file changed, 16 insertions(+), 10 deletions(-) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index 49f51f221f53e..5e53ae90909fa 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -17,30 +17,36 @@ public void TestUnused() static unsafe class C { - static byte Method() + static byte Method(int i) { - return 42; + Console.Write(i); + return 0; } static void Main() { - byte* p = stackalloc byte[3] { 42, Method(), 42 }; + var p = stackalloc[] { 42, Method(1), 42, Method(2) }; } } "; CompileAndVerify(text, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), options: TestOptions.UnsafeReleaseExe, + expectedOutput: "12", verify: Verification.Passes).VerifyIL("C.Main", @"{ - // Code size 10 (0xa) + // Code size 19 (0x13) .maxstack 1 - IL_0000: ldc.i4.3 - IL_0001: conv.u - IL_0002: pop - IL_0003: call ""byte C.Method()"" - IL_0008: pop - IL_0009: ret + IL_0000: ldc.i4.s 16 + IL_0002: conv.u + IL_0003: pop + IL_0004: ldc.i4.1 + IL_0005: call ""byte C.Method(int)"" + IL_000a: pop + IL_000b: ldc.i4.2 + IL_000c: call ""byte C.Method(int)"" + IL_0011: pop + IL_0012: ret }"); } From 1eb959965e6dc115c2c71afe05529e14d44a40ab Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 11:20:50 +0330 Subject: [PATCH 19/43] Assert symbols --- .../Semantic/Semantics/StackAllocInitializerTests.cs | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 9c7fcfa03f559..65a9466d4be6f 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -94,7 +94,12 @@ public void Method1() Assert.Equal("obj1", obj1.Identifier.Text); var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); + Assert.Null(obj1Value.Symbol); Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); + var symbol = model.GetDeclaredSymbol(obj1); + Assert.Equal("System.Double* obj1", symbol.ToTestDisplayString()); } [Fact] @@ -374,6 +379,7 @@ public static implicit operator Test(int* value) Assert.Equal("obj1", obj1.Identifier.Text); var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); + Assert.NotNull(obj1Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); Assert.Equal("Test", obj1Value.ConvertedType.Name); Assert.Equal(ConversionKind.ImplicitUserDefined, obj1Value.ImplicitConversion.Kind); @@ -382,6 +388,7 @@ public static implicit operator Test(int* value) Assert.Equal("obj2", obj2.Identifier.Text); var obj2Value = model.GetSemanticInfoSummary(obj2.Initializer.Value); + Assert.Null(obj2Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.Type).PointedAtType.SpecialType); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.Identity, obj2Value.ImplicitConversion.Kind); @@ -390,6 +397,7 @@ public static implicit operator Test(int* value) Assert.Equal("obj3", obj3.Identifier.Text); var obj3Value = model.GetSemanticInfoSummary(obj3.Initializer.Value); + Assert.Null(obj3Value.Symbol); Assert.Equal("Span", obj3Value.Type.Name); Assert.Equal("Span", obj3Value.ConvertedType.Name); Assert.Equal(ConversionKind.Identity, obj3Value.ImplicitConversion.Kind); @@ -398,6 +406,7 @@ public static implicit operator Test(int* value) Assert.Equal("obj4", obj4.Identifier.Text); var obj4Value = model.GetSemanticInfoSummary(obj4.Initializer.Value); + Assert.Null(obj4Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.Type).PointedAtType.SpecialType); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.Identity, obj4Value.ImplicitConversion.Kind); @@ -406,6 +415,7 @@ public static implicit operator Test(int* value) Assert.Equal("obj5", obj5.Identifier.Text); var obj5Value = model.GetSemanticInfoSummary(obj5.Initializer.Value); + Assert.Null(obj5Value.Symbol); Assert.Null(obj5Value.Type); Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj5Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.NoConversion, obj5Value.ImplicitConversion.Kind); From b2a28ebaaff8b94e1765b1e7dfd4de6ef9f3e8d8 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Thu, 8 Feb 2018 11:27:36 +0330 Subject: [PATCH 20/43] Emit element-wise assignments for anything larger than byte --- .../CodeGen/EmitStackAllocInitializer.cs | 2 +- .../CodeGen/StackAllocInitializerTests.cs | 81 ++++++++++++++----- 2 files changed, 61 insertions(+), 22 deletions(-) diff --git a/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs index 73acd341f9123..fa3e0c33e88fa 100644 --- a/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs +++ b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs @@ -48,7 +48,7 @@ private ArrayInitializerStyle ShouldEmitBlockInitializerForStackAlloc(TypeSymbol elementType = elementType.EnumUnderlyingType(); - if (elementType.SpecialType.IsBlittable()) + if (elementType.SpecialType.SizeInBytes() == 1) { int initCount = 0; int constCount = 0; diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index 5e53ae90909fa..344acb6d7a99c 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -231,17 +231,29 @@ public void TestInt32Pointer() { Test("System.Int32", @"{ - // Code size 21 (0x15) + // Code size 27 (0x1b) .maxstack 4 IL_0000: ldc.i4.s 12 IL_0002: conv.u IL_0003: localloc IL_0005: dup - IL_0006: ldsflda "".__StaticArrayInitTypeSize=12 .E429CCA3F703A39CC5954A6572FEC9086135B34E"" - IL_000b: ldc.i4.s 12 - IL_000d: cpblk - IL_000f: call ""void C.Print(int*)"" - IL_0014: ret + IL_0006: ldc.i4.1 + IL_0007: stind.i4 + IL_0008: dup + IL_0009: ldc.i4.4 + IL_000a: add + IL_000b: ldc.i4.2 + IL_000c: stind.i4 + IL_000d: dup + IL_000e: ldc.i4.2 + IL_000f: conv.i + IL_0010: ldc.i4.4 + IL_0011: mul + IL_0012: add + IL_0013: ldc.i4.3 + IL_0014: stind.i4 + IL_0015: call ""void C.Print(int*)"" + IL_001a: ret }"); } @@ -250,17 +262,32 @@ public void TestInt64Pointer() { Test("System.Int64", @"{ - // Code size 21 (0x15) + // Code size 30 (0x1e) .maxstack 4 IL_0000: ldc.i4.s 24 IL_0002: conv.u IL_0003: localloc IL_0005: dup - IL_0006: ldsflda "".__StaticArrayInitTypeSize=24 .E2D1839ED1706F7D470D87F8C48A5584CAFA5A12"" - IL_000b: ldc.i4.s 24 - IL_000d: cpblk - IL_000f: call ""void C.Print(long*)"" - IL_0014: ret + IL_0006: ldc.i4.1 + IL_0007: conv.i8 + IL_0008: stind.i8 + IL_0009: dup + IL_000a: ldc.i4.8 + IL_000b: add + IL_000c: ldc.i4.2 + IL_000d: conv.i8 + IL_000e: stind.i8 + IL_000f: dup + IL_0010: ldc.i4.2 + IL_0011: conv.i + IL_0012: ldc.i4.8 + IL_0013: mul + IL_0014: add + IL_0015: ldc.i4.3 + IL_0016: conv.i8 + IL_0017: stind.i8 + IL_0018: call ""void C.Print(long*)"" + IL_001d: ret }"); } @@ -343,7 +370,7 @@ public readonly ref struct ReadOnlySpan CompileAndVerify(comp, verify: Verification.Fails, expectedOutput: @"123") .VerifyIL("C.Main", @"{ - // Code size 31 (0x1f) + // Code size 37 (0x25) .maxstack 4 .locals init (int V_0) IL_0000: ldc.i4.3 @@ -354,14 +381,26 @@ .locals init (int V_0) IL_0005: mul.ovf.un IL_0006: localloc IL_0008: dup - IL_0009: ldsflda "".__StaticArrayInitTypeSize=12 .E429CCA3F703A39CC5954A6572FEC9086135B34E"" - IL_000e: ldc.i4.s 12 - IL_0010: cpblk - IL_0012: ldloc.0 - IL_0013: newobj ""System.Span..ctor(void*, int)"" - IL_0018: call ""System.ReadOnlySpan System.ReadOnlySpan.op_Implicit(System.Span)"" - IL_001d: pop - IL_001e: ret + IL_0009: ldc.i4.1 + IL_000a: stind.i4 + IL_000b: dup + IL_000c: ldc.i4.4 + IL_000d: add + IL_000e: ldc.i4.2 + IL_000f: stind.i4 + IL_0010: dup + IL_0011: ldc.i4.2 + IL_0012: conv.i + IL_0013: ldc.i4.4 + IL_0014: mul + IL_0015: add + IL_0016: ldc.i4.3 + IL_0017: stind.i4 + IL_0018: ldloc.0 + IL_0019: newobj ""System.Span..ctor(void*, int)"" + IL_001e: call ""System.ReadOnlySpan System.ReadOnlySpan.op_Implicit(System.Span)"" + IL_0023: pop + IL_0024: ret }"); } From c13dd3ae6c32b068cd6120b0464e4df1fb467876 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Feb 2018 00:57:34 +0330 Subject: [PATCH 21/43] Revert code. --- .../Semantic/Semantics/StackAllocInitializerTests.cs | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 65a9466d4be6f..9c7fcfa03f559 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -94,12 +94,7 @@ public void Method1() Assert.Equal("obj1", obj1.Identifier.Text); var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); - Assert.Null(obj1Value.Symbol); Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); - Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.ConvertedType).PointedAtType.SpecialType); - Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); - var symbol = model.GetDeclaredSymbol(obj1); - Assert.Equal("System.Double* obj1", symbol.ToTestDisplayString()); } [Fact] @@ -379,7 +374,6 @@ public static implicit operator Test(int* value) Assert.Equal("obj1", obj1.Identifier.Text); var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); - Assert.NotNull(obj1Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); Assert.Equal("Test", obj1Value.ConvertedType.Name); Assert.Equal(ConversionKind.ImplicitUserDefined, obj1Value.ImplicitConversion.Kind); @@ -388,7 +382,6 @@ public static implicit operator Test(int* value) Assert.Equal("obj2", obj2.Identifier.Text); var obj2Value = model.GetSemanticInfoSummary(obj2.Initializer.Value); - Assert.Null(obj2Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.Type).PointedAtType.SpecialType); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj2Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.Identity, obj2Value.ImplicitConversion.Kind); @@ -397,7 +390,6 @@ public static implicit operator Test(int* value) Assert.Equal("obj3", obj3.Identifier.Text); var obj3Value = model.GetSemanticInfoSummary(obj3.Initializer.Value); - Assert.Null(obj3Value.Symbol); Assert.Equal("Span", obj3Value.Type.Name); Assert.Equal("Span", obj3Value.ConvertedType.Name); Assert.Equal(ConversionKind.Identity, obj3Value.ImplicitConversion.Kind); @@ -406,7 +398,6 @@ public static implicit operator Test(int* value) Assert.Equal("obj4", obj4.Identifier.Text); var obj4Value = model.GetSemanticInfoSummary(obj4.Initializer.Value); - Assert.Null(obj4Value.Symbol); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.Type).PointedAtType.SpecialType); Assert.Equal(SpecialType.System_Int32, ((PointerTypeSymbol)obj4Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.Identity, obj4Value.ImplicitConversion.Kind); @@ -415,7 +406,6 @@ public static implicit operator Test(int* value) Assert.Equal("obj5", obj5.Identifier.Text); var obj5Value = model.GetSemanticInfoSummary(obj5.Initializer.Value); - Assert.Null(obj5Value.Symbol); Assert.Null(obj5Value.Type); Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj5Value.ConvertedType).PointedAtType.SpecialType); Assert.Equal(ConversionKind.NoConversion, obj5Value.ImplicitConversion.Kind); From 9c5a72e5f3b2ce34618e7510e803bcbe671621b8 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Feb 2018 01:01:05 +0330 Subject: [PATCH 22/43] Clean up. --- .../Test/Semantic/Semantics/StackAllocInitializerTests.cs | 5 ----- 1 file changed, 5 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 9c7fcfa03f559..fbc14718218b4 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -1,14 +1,10 @@ // Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. -using System; -using System.Collections.Generic; using System.Linq; -using System.Threading; using Microsoft.CodeAnalysis.CSharp.Symbols; using Microsoft.CodeAnalysis.CSharp.Syntax; using Microsoft.CodeAnalysis.CSharp.Test.Utilities; using Microsoft.CodeAnalysis.Test.Utilities; -using Roslyn.Test.Utilities; using Xunit; namespace Microsoft.CodeAnalysis.CSharp.UnitTests @@ -34,7 +30,6 @@ public void Method() var p5 = stackalloc[] { (1, null) }; var p6 = stackalloc[] { () => { } }; var p7 = stackalloc[] { new {} , new { i = 0 } }; - } } namespace System { From 442c0d7f6216541a7ba34efbdd6c4412ddb2e9fe Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Feb 2018 01:04:52 +0330 Subject: [PATCH 23/43] Add braces. --- .../CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs b/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs index a3986c4091ce6..5ea5170596d6f 100644 --- a/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs +++ b/src/Compilers/CSharp/Portable/FlowAnalysis/PreciseAbstractFlowPass.cs @@ -2580,7 +2580,9 @@ public override BoundNode VisitStackAllocArrayCreation(BoundStackAllocArrayCreat if (node.InitializerOpt != null && !node.InitializerOpt.Initializers.IsDefault) { foreach (var element in node.InitializerOpt.Initializers) + { VisitRvalue(element); + } } if (_trackExceptions) NotePossibleException(node); From 779ff1803eceb58de23d5d7111be3f5414be596a Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Feb 2018 01:31:07 +0330 Subject: [PATCH 24/43] Assert symbol from syntax --- .../Test/Semantic/Semantics/StackAllocInitializerTests.cs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index fbc14718218b4..e75a4ee5b6d80 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -90,6 +90,10 @@ public void Method1() var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); + Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.ConvertedType).PointedAtType.SpecialType); + Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); + var declared = model.GetDeclaredSymbol(obj1.Initializer.Value); + Assert.Null(declared); } [Fact] From 0f92e1d02b1b3feadde0edcbdb759a71ba5dcabd Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Feb 2018 15:26:21 +0330 Subject: [PATCH 25/43] Share code. --- .../Portable/Binder/Binder_Expressions.cs | 19 +++++++------------ 1 file changed, 7 insertions(+), 12 deletions(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index f4cb3d8b7e356..a46e21c6a21ff 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -2718,9 +2718,6 @@ private BoundExpression BindImplicitArrayCreationExpression(ImplicitArrayCreatio private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitStackAllocArrayCreationExpressionSyntax node, DiagnosticBag diagnostics) { - bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); - bool hasErrors = !inLegalPosition; - InitializerExpressionSyntax initializer = node.Initializer; ImmutableArray boundInitializerExpressions = BindArrayInitializerExpressions(initializer, diagnostics, dimension: 1, rank: 1); @@ -2742,11 +2739,11 @@ private BoundExpression BindImplicitStackAllocArrayCreationExpression(ImplicitSt return BindStackAllocWithInitializer( node, initializer, - type: inLegalPosition ? GetStackAllocType(node, bestType, diagnostics) : null, + type: GetStackAllocType(node, bestType, diagnostics, out bool hasErrors), elementType: bestType, sizeOpt: null, diagnostics, - hasErrors, + hasErrors: hasErrors, boundInitializerExpressions); } @@ -3068,9 +3065,6 @@ private BoundArrayCreation BindArrayCreationWithInitializer( private BoundExpression BindStackAllocArrayCreationExpression( StackAllocArrayCreationExpressionSyntax node, DiagnosticBag diagnostics) { - bool inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); - bool hasErrors = !inLegalPosition; - TypeSyntax typeSyntax = node.Type; if (typeSyntax.Kind() != SyntaxKind.ArrayType) @@ -3089,6 +3083,7 @@ private BoundExpression BindStackAllocArrayCreationExpression( TypeSyntax elementTypeSyntax = arrayTypeSyntax.ElementType; TypeSymbol elementType = BindType(elementTypeSyntax, diagnostics); + TypeSymbol type = GetStackAllocType(node, elementType, diagnostics, out bool hasErrors); if (!elementType.IsErrorType() && elementType.IsManagedType) { Error(diagnostics, ErrorCode.ERR_ManagedAddr, elementTypeSyntax, elementType); @@ -3126,8 +3121,6 @@ private BoundExpression BindStackAllocArrayCreationExpression( new PointerTypeSymbol(elementType)); } - TypeSymbol type = inLegalPosition ? GetStackAllocType(node, elementType, diagnostics) : null; - ExpressionSyntax countSyntax = rankSpecifiers[0].Sizes[0]; BoundExpression count = null; if (countSyntax.Kind() != SyntaxKind.OmittedArraySizeExpression) @@ -3178,9 +3171,11 @@ private bool ReportBadStackAllocPosition(SyntaxNode node, DiagnosticBag diagnost return inLegalPosition; } - private TypeSymbol GetStackAllocType(SyntaxNode node, TypeSymbol elementType, DiagnosticBag diagnostics) + private TypeSymbol GetStackAllocType(SyntaxNode node, TypeSymbol elementType, DiagnosticBag diagnostics, out bool hasErrors) { - if (!node.IsVariableDeclarationInitialization()) + var inLegalPosition = ReportBadStackAllocPosition(node, diagnostics); + hasErrors = !inLegalPosition; + if (inLegalPosition && !node.IsVariableDeclarationInitialization()) { CheckFeatureAvailability(node, MessageID.IDS_FeatureRefStructs, diagnostics); GetWellKnownTypeMember(Compilation, WellKnownMember.System_Span_T__ctor, diagnostics, syntax: node); From 1f8a00fd52f5964b3f03483d257d2afbca0db46b Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Sun, 18 Feb 2018 00:10:55 +0330 Subject: [PATCH 26/43] Use initblk whenever possible regardless of type size --- .../CodeGen/EmitStackAllocInitializer.cs | 19 ++++++++++--- .../CodeGen/StackAllocInitializerTests.cs | 28 +++++++++---------- .../Core/Portable/CodeGen/ILBuilderEmit.cs | 4 +-- 3 files changed, 31 insertions(+), 20 deletions(-) diff --git a/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs index fa3e0c33e88fa..bf650097be959 100644 --- a/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs +++ b/src/Compilers/CSharp/Portable/CodeGen/EmitStackAllocInitializer.cs @@ -30,11 +30,22 @@ private void EmitStackAllocInitializers(TypeSymbol type, BoundArrayInitializatio else { ImmutableArray data = this.GetRawData(initExprs); - _builder.EmitStackAllocBlockInitializer(data, inits.Syntax, _diagnostics); + if (data.All(datum => datum == data[0])) + { + _builder.EmitStackAllocBlockInitializer(data, inits.Syntax, emitInitBlock: true, _diagnostics); + } + else if (elementType.SpecialType.SizeInBytes() == 1) + { + _builder.EmitStackAllocBlockInitializer(data, inits.Syntax, emitInitBlock: false, _diagnostics); - if (initializationStyle == ArrayInitializerStyle.Mixed) + if (initializationStyle == ArrayInitializerStyle.Mixed) + { + EmitElementStackAllocInitializers(elementType, initExprs, includeConstants: false); + } + } + else { - EmitElementStackAllocInitializers(elementType, initExprs, includeConstants: false); + EmitElementStackAllocInitializers(elementType, initExprs, includeConstants: true); } } } @@ -48,7 +59,7 @@ private ArrayInitializerStyle ShouldEmitBlockInitializerForStackAlloc(TypeSymbol elementType = elementType.EnumUnderlyingType(); - if (elementType.SpecialType.SizeInBytes() == 1) + if (elementType.SpecialType.IsBlittable()) { int initCount = 0; int constCount = 0; diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index 344acb6d7a99c..a31a6d1b67c50 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -58,15 +58,15 @@ public void TestIdenticalBytes() static unsafe class C { - static void Print(byte* p) + static void Print(uint* p) { for (int i = 0; i < 3; i++) - Console.Write(p[i]); + Console.Write(p[i].ToString(""x"")); } static void Main() { - byte* p = stackalloc byte[3] { 42, 42, 42 }; + var p = stackalloc[] { 0xffffffff, 0xffffffff, 0xffffffff }; Print(p); } } @@ -74,19 +74,19 @@ static void Main() CompileAndVerify(text, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), options: TestOptions.UnsafeReleaseExe, - verify: Verification.Fails, expectedOutput: @"424242").VerifyIL("C.Main", + verify: Verification.Fails, expectedOutput: @"ffffffffffffffffffffffff").VerifyIL("C.Main", @"{ - // Code size 16 (0x10) + // Code size 21 (0x15) .maxstack 4 - IL_0000: ldc.i4.3 - IL_0001: conv.u - IL_0002: localloc - IL_0004: dup - IL_0005: ldc.i4.s 42 - IL_0007: ldc.i4.3 - IL_0008: initblk - IL_000a: call ""void C.Print(byte*)"" - IL_000f: ret + IL_0000: ldc.i4.s 12 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: ldc.i4 0xff + IL_000b: ldc.i4.s 12 + IL_000d: initblk + IL_000f: call ""void C.Print(uint*)"" + IL_0014: ret }"); } diff --git a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs index e367e730aae66..62e17afaa5ff5 100644 --- a/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs +++ b/src/Compilers/Core/Portable/CodeGen/ILBuilderEmit.cs @@ -88,9 +88,9 @@ internal void EmitArrayBlockInitializer(ImmutableArray data, SyntaxNode sy EmitToken(initializeArray, syntaxNode, diagnostics); } - internal void EmitStackAllocBlockInitializer(ImmutableArray data, SyntaxNode syntaxNode, DiagnosticBag diagnostics) + internal void EmitStackAllocBlockInitializer(ImmutableArray data, SyntaxNode syntaxNode, bool emitInitBlock, DiagnosticBag diagnostics) { - if (data.All(datum => datum == data[0])) + if (emitInitBlock) { // All bytes are the same, no need for metadata blob From 5836ddaeb9a52834f99ccb29d7fa07ba942ff40d Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Sun, 18 Feb 2018 01:30:59 +0330 Subject: [PATCH 27/43] Address test failures --- .../Test/Syntax/Parsing/ParserErrorMessageTests.cs | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/ParserErrorMessageTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/ParserErrorMessageTests.cs index 0c86ebfeab5e9..c853a5831a720 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/ParserErrorMessageTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/ParserErrorMessageTests.cs @@ -4532,19 +4532,16 @@ unsafe public static int Main() // Extra errors CreateStandardCompilation(test, options: TestOptions.UnsafeDebugDll).VerifyDiagnostics( // (7,34): error CS1002: ; expected - // int *pp = stackalloc int 30; + // int *pp = stackalloc int 30; Diagnostic(ErrorCode.ERR_SemicolonExpected, "30").WithLocation(7, 34), - // (6,18): error CS1525: Invalid expression term 'stackalloc' - // int *p = stackalloc int (30); - Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(6, 18), // (6,29): error CS1575: A stackalloc expression requires [] after type - // int *p = stackalloc int (30); + // int *p = stackalloc int (30); Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int").WithLocation(6, 29), // (7,30): error CS1575: A stackalloc expression requires [] after type - // int *pp = stackalloc int 30; + // int *pp = stackalloc int 30; Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int").WithLocation(7, 30), // (7,34): error CS0201: Only assignment, call, increment, decrement, and new object expressions can be used as a statement - // int *pp = stackalloc int 30; + // int *pp = stackalloc int 30; Diagnostic(ErrorCode.ERR_IllegalStatement, "30").WithLocation(7, 34) ); } From 8e467f4371a16d6b148e65d1a8cefef7aedc187b Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Wed, 21 Feb 2018 21:34:13 +0330 Subject: [PATCH 28/43] PR feedback --- .../CodeGen/StackAllocInitializerTests.cs | 75 ++++++++++++++++++- 1 file changed, 74 insertions(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index a31a6d1b67c50..3874cb9f6b543 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -51,7 +51,80 @@ .maxstack 1 } [Fact] - public void TestIdenticalBytes() + public void TestEmpty() + { + var text = @" +using System; + +static unsafe class C +{ + static void Use(int* p) + { + } + + static void Main() + { + var p = stackalloc int[] {}; + Use(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails).VerifyIL("C.Main", +@"{ + // Code size 8 (0x8) + .maxstack 1 + IL_0000: ldc.i4.0 + IL_0001: conv.u + IL_0002: call ""void C.Use(int*)"" + IL_0007: ret +}"); + } + + [Fact] + public void TestIdenticalBytes1() + { + var text = @" +using System; + +static unsafe class C +{ + static void Print(byte* p) + { + for (int i = 0; i < 3; i++) + Console.Write(p[i]); + } + + static void Main() + { + byte* p = stackalloc byte[3] { 42, 42, 42 }; + Print(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails, expectedOutput: @"424242").VerifyIL("C.Main", +@"{ + // Code size 16 (0x10) + .maxstack 4 + IL_0000: ldc.i4.3 + IL_0001: conv.u + IL_0002: localloc + IL_0004: dup + IL_0005: ldc.i4.s 42 + IL_0007: ldc.i4.3 + IL_0008: initblk + IL_000a: call ""void C.Print(byte*)"" + IL_000f: ret +}"); + } + + [Fact] + public void TestIdenticalBytes2() { var text = @" using System; From 130659f3d80e3bb631e573ad913e38a7e446b2f9 Mon Sep 17 00:00:00 2001 From: Vladimir Sadov Date: Wed, 21 Feb 2018 21:07:12 -0800 Subject: [PATCH 29/43] Create testplan doc for stackalloc initializers. --- .../testplans/stackalloc_initializers.md | 61 +++++++++++++++++++ 1 file changed, 61 insertions(+) create mode 100644 docs/compilers/CSharp/testplans/stackalloc_initializers.md diff --git a/docs/compilers/CSharp/testplans/stackalloc_initializers.md b/docs/compilers/CSharp/testplans/stackalloc_initializers.md new file mode 100644 index 0000000000000..170bd366e8f76 --- /dev/null +++ b/docs/compilers/CSharp/testplans/stackalloc_initializers.md @@ -0,0 +1,61 @@ + +Here is the rough testplan for the stackalloc initializers feature. + +Some tests may already exist or be part of other tests. +In such case we can just check them off. + + +## Evaluation semantics ## + +### correctness ### +- [ ] check that partial result is not visible. - if element throws and exception caught, whole thing is not reassigned to the new value. (try spans and ordinary stackallocs) +- [ ] if we refer to the elements of outer local in the initializers we still see the old values. +- [ ] put `await` in the size or in the middle of element initializers +- [ ] initializer uses a local that is captured by a lambda expression. See that if lambda changes the value of the local, then we see the updated value when initializing + +### errors ## +- [x] invalid array shapes - multidimensional, nested, size mismatches, ... +- [x] bad conversions +- [x] bad local initialization contexts - using, fixed. +- [ ] should it work in `for(int* x = stackalloc...)`, probably the same ways as before... ? +- [X] missing various span parts. +- [ ] outer expression is not usable in elements (not assigned yet) +- [ ] in the array type inference case the outer expression cannot be used to determine the type - [ ] should be an error and should not be some kind of crash due to circular dependency +- [ ] make it to infer strange types. + - [ ] dynamic + - [ ] ref-struct + - [ ] void (use a void method to initialize an element) + - [X] lambda expression (formally typeless) + - [X] null + - [ ] discard `_` - is not even a value. + +## API ## + +- [ ] GetSymbolInfo + - [X] on the whole node (checked for ordinary int* and Span) + - [ ] on size expression (including cases when implicit conversions happen) + - [ ] on element expressions (including cases when implicit conversions happen) + +- [ ] GetDeclaredSymbol + - [ ] the expression does not declare anything, try with whole node, elements, just in case.. + + + +## Manual IDE testing ## +The more detailed plan is here https://github.com/dotnet/roslyn/wiki/Manual-Testing although many items may end up N/A. +We will have to go through the list before merging. + +This may not be possible to be done with just public bits - I am not sure. Will check myself + + +Preliminary - the scenarios of concern are not many, since this is an expression and fairly simple too. + +- [ ] extract whole thing to a method (regardless whether that works for stackallocks, it should behave rationally, even if resulting code has errors due to escape rules, etc...) +- [ ] extract array size and elements to a method, especially if there are implicit conversions. (should just work if semantic APIs work correctly) +- [ ] general typing of the construct + - [ ] IDE should not autocorrect into something meaningless + - [ ] inside the initializer the dropdowns shoudl be as expected - i.e. variables in scope + - [ ] autocompletion of parens and braces - [] { } + + + From 82f521cdc8dab64483962ae0e0671c82631663c1 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Tue, 27 Feb 2018 23:45:00 +0330 Subject: [PATCH 30/43] [stackalloc-init] Improve error tolerance when parsing invalid implicit stackalloc arrays (#24914) --- .../Portable/CSharpResources.Designer.cs | 9 + .../CSharp/Portable/CSharpResources.resx | 3 + .../CSharp/Portable/Errors/ErrorCode.cs | 1 + .../CSharp/Portable/Parser/LanguageParser.cs | 20 ++ .../Portable/xlf/CSharpResources.cs.xlf | 5 + .../Portable/xlf/CSharpResources.de.xlf | 5 + .../Portable/xlf/CSharpResources.es.xlf | 5 + .../Portable/xlf/CSharpResources.fr.xlf | 5 + .../Portable/xlf/CSharpResources.it.xlf | 5 + .../Portable/xlf/CSharpResources.ja.xlf | 5 + .../Portable/xlf/CSharpResources.ko.xlf | 5 + .../Portable/xlf/CSharpResources.pl.xlf | 5 + .../Portable/xlf/CSharpResources.pt-BR.xlf | 5 + .../Portable/xlf/CSharpResources.ru.xlf | 5 + .../Portable/xlf/CSharpResources.tr.xlf | 5 + .../Portable/xlf/CSharpResources.zh-Hans.xlf | 5 + .../Portable/xlf/CSharpResources.zh-Hant.xlf | 5 + .../Semantics/StackAllocInitializerTests.cs | 4 + .../Parsing/StackAllocInitializerTests.cs | 281 ++++++++++++++++-- 19 files changed, 355 insertions(+), 28 deletions(-) diff --git a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs index a0980827131e8..9f55005b13f6b 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs +++ b/src/Compilers/CSharp/Portable/CSharpResources.Designer.cs @@ -5983,6 +5983,15 @@ internal static string ERR_InvalidSpecifier { } } + /// + /// Looks up a localized string similar to "Invalid rank specifier: expected ']'. + /// + internal static string ERR_InvalidStackAllocArray { + get { + return ResourceManager.GetString("ERR_InvalidStackAllocArray", resourceCulture); + } + } + /// /// Looks up a localized string similar to Invalid version {0} for /subsystemversion. The version must be 6.02 or greater for ARM or AppContainerExe, and 4.00 or greater otherwise. /// diff --git a/src/Compilers/CSharp/Portable/CSharpResources.resx b/src/Compilers/CSharp/Portable/CSharpResources.resx index e2e8c61b0d2ad..85d29f08219db 100644 --- a/src/Compilers/CSharp/Portable/CSharpResources.resx +++ b/src/Compilers/CSharp/Portable/CSharpResources.resx @@ -5256,4 +5256,7 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/Errors/ErrorCode.cs b/src/Compilers/CSharp/Portable/Errors/ErrorCode.cs index 61e0310af8e86..7ec6a195b0e23 100644 --- a/src/Compilers/CSharp/Portable/Errors/ErrorCode.cs +++ b/src/Compilers/CSharp/Portable/Errors/ErrorCode.cs @@ -1556,6 +1556,7 @@ internal enum ErrorCode ERR_FeatureNotAvailableInVersion7_3 = 8370, WRN_AttributesOnBackingFieldsNotAvailable = 8371, ERR_DoNotUseFixedBufferAttrOnProperty = 8372, + ERR_InvalidStackAllocArray = 8373, // Note: you will need to re-generate compiler code after adding warnings (build\scripts\generate-compiler-code.cmd) } diff --git a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs index 6043e070a35cb..144321c3c1998 100644 --- a/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs +++ b/src/Compilers/CSharp/Portable/Parser/LanguageParser.cs @@ -10703,6 +10703,26 @@ private ExpressionSyntax ParseImplicitlyTypedStackAllocExpression() var @stackalloc = this.EatToken(SyntaxKind.StackAllocKeyword); @stackalloc = CheckFeatureAvailability(@stackalloc, MessageID.IDS_FeatureStackAllocInitializer); var openBracket = this.EatToken(SyntaxKind.OpenBracketToken); + + int lastTokenPosition = -1; + while (IsMakingProgress(ref lastTokenPosition)) + { + if (this.IsPossibleExpression()) + { + var size = this.AddError(this.ParseExpressionCore(), ErrorCode.ERR_InvalidStackAllocArray); + openBracket = AddTrailingSkippedSyntax(openBracket, size); + } + + if (this.CurrentToken.Kind == SyntaxKind.CommaToken) + { + var comma = this.AddError(this.EatToken(), ErrorCode.ERR_InvalidStackAllocArray); + openBracket = AddTrailingSkippedSyntax(openBracket, comma); + continue; + } + + break; + } + var closeBracket = this.EatToken(SyntaxKind.CloseBracketToken); var initializer = this.ParseArrayInitializer(); return _syntaxFactory.ImplicitStackAllocArrayCreationExpression(@stackalloc, openBracket, closeBracket, initializer); diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf index b589548dce920..19c3d9e4ea002 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf @@ -8615,6 +8615,11 @@ Pokud chcete odstranit toto varování, můžete místo toho použít /reference stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf index 1dcb27e2caed8..7402b2b2c6cba 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf @@ -8615,6 +8615,11 @@ Um die Warnung zu beheben, können Sie stattdessen /reference verwenden (Einbett stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf index 5765b6ad8490a..52e2c14aa86c9 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf @@ -8615,6 +8615,11 @@ Para eliminar la advertencia puede usar /reference (establezca la propiedad Embe stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf index dad97f8fed2f7..6efe2f708df26 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf @@ -8615,6 +8615,11 @@ Pour supprimer l'avertissement, vous pouvez utiliser la commande /reference (dé stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf index 0448ebf73ddf4..3cabf43c022e0 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf @@ -8615,6 +8615,11 @@ Per rimuovere l'avviso, è invece possibile usare /reference (impostare la propr stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf index 904a3e6402123..09716d9fd29f9 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf @@ -8615,6 +8615,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf index 873df201acb85..08a206c046a22 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf @@ -8615,6 +8615,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf index 01f7c3621a3e7..8133e335f8d3e 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf @@ -8615,6 +8615,11 @@ Aby usunąć ostrzeżenie, możesz zamiast tego użyć opcji /reference (ustaw w stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf index 7dd155d15d83d..b7b704f1e0a42 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf @@ -8615,6 +8615,11 @@ Para incorporar informações de tipo de interoperabilidade para os dois assembl stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf index 8132882ecad60..a9a321ed0d53a 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf @@ -8615,6 +8615,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf index 68bd6dd346d8f..805571e28c946 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf @@ -8615,6 +8615,11 @@ Uyarıyı kaldırmak için, /reference kullanabilirsiniz (Birlikte Çalışma T stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf index a7a3e6b52c4b5..0f0afa50be02e 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf @@ -8615,6 +8615,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf index 5d48d59a502e0..71d91c5f1e196 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf @@ -8615,6 +8615,11 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ stackalloc initializer + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index e75a4ee5b6d80..dfa1c4483c0bb 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -248,10 +248,14 @@ unsafe class Test public void Method1() { var obj1 = stackalloc int[,] { 1 }; + var obj2 = stackalloc [,] { 1 }; } }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( + // (7,35): error CS8373: "Invalid rank specifier: expected ']' + // var obj2 = stackalloc [,] { 1 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(7, 35), // (6,31): error CS1575: A stackalloc expression requires [] after type // var obj1 = stackalloc int[,] { 1 }; Diagnostic(ErrorCode.ERR_BadStackAllocExpr, "int[,]").WithLocation(6, 31) diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs index e586f8a56e03f..22e71e0949d51 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs @@ -116,47 +116,272 @@ public void StackAllocInitializer_04() // (1,1): error CS8107: Feature 'stackalloc initializer' is not available in C# 7.0. Please use language version 7.3 or greater. // stackalloc[1] { 42 } Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1), - // (1,12): error CS1003: Syntax error, ']' expected + // (1,12): error CS8373: "Invalid rank specifier: expected ']' // stackalloc[1] { 42 } - Diagnostic(ErrorCode.ERR_SyntaxError, "1").WithArguments("]", "").WithLocation(1, 12), - // (1,12): error CS1514: { expected - // stackalloc[1] { 42 } - Diagnostic(ErrorCode.ERR_LbraceExpected, "1").WithLocation(1, 12), - // (1,13): error CS1003: Syntax error, ',' expected - // stackalloc[1] { 42 } - Diagnostic(ErrorCode.ERR_SyntaxError, "]").WithArguments(",", "]").WithLocation(1, 13), - // (1,15): error CS1003: Syntax error, ',' expected - // stackalloc[1] { 42 } - Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments(",", "{").WithLocation(1, 15), - // (1,21): error CS1513: } expected - // stackalloc[1] { 42 } - Diagnostic(ErrorCode.ERR_RbraceExpected, "").WithLocation(1, 21)); + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "1").WithLocation(1, 12) + ); N(SyntaxKind.ImplicitStackAllocArrayCreationExpression); { N(SyntaxKind.StackAllocKeyword); N(SyntaxKind.OpenBracketToken); - M(SyntaxKind.CloseBracketToken); + N(SyntaxKind.CloseBracketToken); N(SyntaxKind.ArrayInitializerExpression); { - M(SyntaxKind.OpenBraceToken); + N(SyntaxKind.OpenBraceToken); N(SyntaxKind.NumericLiteralExpression); { - N(SyntaxKind.NumericLiteralToken, "1"); - } - M(SyntaxKind.CommaToken); - N(SyntaxKind.ArrayInitializerExpression); - { - N(SyntaxKind.OpenBraceToken); - N(SyntaxKind.NumericLiteralExpression); - { - N(SyntaxKind.NumericLiteralToken, "42"); - } - N(SyntaxKind.CloseBraceToken); + N(SyntaxKind.NumericLiteralToken, "42"); } - M(SyntaxKind.CloseBraceToken); + N(SyntaxKind.CloseBraceToken); } } EOF(); } + + [Fact] + public void StackAllocInitializer_05() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28) + ); + } + + [Fact] + public void StackAllocInitializer_06() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3,] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29) + ); + } + + [Fact] + public void StackAllocInitializer_07() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[,3] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28) + ); + } + + [Fact] + public void StackAllocInitializer_08() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[,3 { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3 { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3 { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), + // (4,31): error CS1003: Syntax error, ']' expected + // var x = stackalloc[,3 { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments("]", "{").WithLocation(4, 31) + ); + } + + [Fact] + public void StackAllocInitializer_09() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3 { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3 { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), + // (4,30): error CS1003: Syntax error, ']' expected + // var x = stackalloc[3 { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments("]", "{").WithLocation(4, 30) + ); + } + + [Fact] + public void StackAllocInitializer_10() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3, { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3, { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3, { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), + // (4,31): error CS1003: Syntax error, ']' expected + // var x = stackalloc[3, { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments("]", "{").WithLocation(4, 31) + ); + } + + [Fact] + public void StackAllocInitializer_11() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3,,] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), + // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) + ); + } + + [Fact] + public void StackAllocInitializer_12() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[,3,] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), + // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,3,] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) + ); + } + + [Fact] + public void StackAllocInitializer_13() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[,,3] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 30), + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29) + ); + } + + [Fact] + public void StackAllocInitializer_14() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[3,,3] { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), + // (4,31): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 31), + // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), + // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // var x = stackalloc[3,,3] { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) + ); + } + + [Fact] + public void StackAllocInitializer_15() + { + var test = @" +class C { + void Goo() { + var x = stackalloc[ { 1, 2, 3 }; + } +} +"; + + ParseAndValidate(test, + // (4,29): error CS1003: Syntax error, ']' expected + // var x = stackalloc[ { 1, 2, 3 }; + Diagnostic(ErrorCode.ERR_SyntaxError, "{").WithArguments("]", "{").WithLocation(4, 29) + ); + } } } From 1e681a5bb3a67fd03c2e682bd15bcf2d9469f76a Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Tue, 27 Feb 2018 23:52:54 +0330 Subject: [PATCH 31/43] Add more tests. --- .../testplans/stackalloc_initializers.md | 12 +- .../Portable/Binder/Binder_Expressions.cs | 21 +- .../CodeGen/StackAllocInitializerTests.cs | 40 ++ .../Semantics/StackAllocInitializerTests.cs | 392 ++++++++++++++++-- 4 files changed, 425 insertions(+), 40 deletions(-) diff --git a/docs/compilers/CSharp/testplans/stackalloc_initializers.md b/docs/compilers/CSharp/testplans/stackalloc_initializers.md index 170bd366e8f76..bed611dca773c 100644 --- a/docs/compilers/CSharp/testplans/stackalloc_initializers.md +++ b/docs/compilers/CSharp/testplans/stackalloc_initializers.md @@ -10,24 +10,24 @@ In such case we can just check them off. ### correctness ### - [ ] check that partial result is not visible. - if element throws and exception caught, whole thing is not reassigned to the new value. (try spans and ordinary stackallocs) - [ ] if we refer to the elements of outer local in the initializers we still see the old values. -- [ ] put `await` in the size or in the middle of element initializers +- [x] put `await` in the size or in the middle of element initializers - [ ] initializer uses a local that is captured by a lambda expression. See that if lambda changes the value of the local, then we see the updated value when initializing ### errors ## - [x] invalid array shapes - multidimensional, nested, size mismatches, ... - [x] bad conversions - [x] bad local initialization contexts - using, fixed. -- [ ] should it work in `for(int* x = stackalloc...)`, probably the same ways as before... ? +- [x] should it work in `for(int* x = stackalloc...)`, probably the same ways as before... ? - [X] missing various span parts. -- [ ] outer expression is not usable in elements (not assigned yet) +- [x] outer expression is not usable in elements (not assigned yet) - [ ] in the array type inference case the outer expression cannot be used to determine the type - [ ] should be an error and should not be some kind of crash due to circular dependency - [ ] make it to infer strange types. - - [ ] dynamic + - [x] dynamic - [ ] ref-struct - - [ ] void (use a void method to initialize an element) + - [x] void (use a void method to initialize an element) - [X] lambda expression (formally typeless) - [X] null - - [ ] discard `_` - is not even a value. + - [x] discard `_` - is not even a value. ## API ## diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs index a46e21c6a21ff..816d0e5bc02e4 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs @@ -3209,16 +3209,19 @@ private BoundExpression BindStackAllocWithInitializer( if (sizeOpt != null) { - int? constantSizeOpt = GetIntegerConstantForArraySize(sizeOpt); - if (!sizeOpt.HasAnyErrors && constantSizeOpt == null) + if (!sizeOpt.HasAnyErrors) { - Error(diagnostics, ErrorCode.ERR_ConstantExpected, sizeOpt.Syntax); - hasErrors = true; - } - else if (boundInitExprOpt.Length != constantSizeOpt) - { - Error(diagnostics, ErrorCode.ERR_ArrayInitializerIncorrectLength, node, constantSizeOpt.Value); - hasErrors = true; + int? constantSizeOpt = GetIntegerConstantForArraySize(sizeOpt); + if (constantSizeOpt == null) + { + Error(diagnostics, ErrorCode.ERR_ConstantExpected, sizeOpt.Syntax); + hasErrors = true; + } + else if (boundInitExprOpt.Length != constantSizeOpt) + { + Error(diagnostics, ErrorCode.ERR_ArrayInitializerIncorrectLength, node, constantSizeOpt.Value); + hasErrors = true; + } } } else diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index 3874cb9f6b543..47d9aaf93318b 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -9,6 +9,46 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests [CompilerTrait(CompilerFeature.StackAllocInitializer)] public class StackAllocInitializerTests : CompilingTestBase { + [Fact] + public void TestElementThrow() + { + var text = @" +using System; + +static unsafe class C +{ + static void Use(int* i) {} + + static int M() { return 0; } + + static void Main() + { + var p = stackalloc[] { M(), true ? throw null : M(), M() }; + Use(p); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + verify: Verification.Fails).VerifyIL("C.Main", +@"{ + // Code size 17 (0x11) + .maxstack 4 + IL_0000: ldc.i4.s 12 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: call ""int C.M()"" + IL_000b: stind.i4 + IL_000c: dup + IL_000d: ldc.i4.4 + IL_000e: add + IL_000f: ldnull + IL_0010: throw +}"); + } + [Fact] public void TestUnused() { diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index dfa1c4483c0bb..533eefb85e32d 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -13,7 +13,7 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests public class StackAllocInitializerTests : CompilingTestBase { [Fact] - public void NoBestType() + public void NoBestType_Pointer() { var comp = CreateCompilationWithMscorlibAndSpan(@" unsafe class Test @@ -21,16 +21,20 @@ unsafe class Test struct A {} struct B {} - public void Method() + void Method(dynamic d, RefStruct r) { - var p1 = stackalloc[] { new A(), new B() }; - var p2 = stackalloc[] { }; - var p3 = stackalloc[] { Method() }; - var p4 = stackalloc[] { null }; - var p5 = stackalloc[] { (1, null) }; - var p6 = stackalloc[] { () => { } }; - var p7 = stackalloc[] { new {} , new { i = 0 } }; + var p0 = stackalloc[] { new A(), new B() }; + var p1 = stackalloc[] { }; + var p2 = stackalloc[] { VoidMethod() }; + var p3 = stackalloc[] { null }; + var p4 = stackalloc[] { (1, null) }; + var p5 = stackalloc[] { () => { } }; + var p6 = stackalloc[] { new {} , new { i = 0 } }; + var p7 = stackalloc[] { d }; + var p8 = stackalloc[] { _ }; } + + public void VoidMethod() {} } namespace System { public struct ValueTuple { @@ -40,32 +44,164 @@ public ValueTuple(T1 a, T2 b) { } ", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( + // void Method(dynamic d, RefStruct r) + Diagnostic(ErrorCode.ERR_SingleTypeNameNotFound, "RefStruct").WithArguments("RefStruct").WithLocation(7, 28), // (9,18): error CS0826: No best type found for implicitly-typed array - // var p1 = stackalloc[] { new A(), new B() }; + // var p0 = stackalloc[] { new A(), new B() }; Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new A(), new B() }").WithLocation(9, 18), // (10,18): error CS0826: No best type found for implicitly-typed array - // var p2 = stackalloc[] { }; + // var p1 = stackalloc[] { }; Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { }").WithLocation(10, 18), // (11,18): error CS0826: No best type found for implicitly-typed array - // var p3 = stackalloc[] { Method() }; - Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { Method() }").WithLocation(11, 18), + // var p2 = stackalloc[] { VoidMethod() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { VoidMethod() }").WithLocation(11, 18), // (12,18): error CS0826: No best type found for implicitly-typed array - // var p4 = stackalloc[] { null }; + // var p3 = stackalloc[] { null }; Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { null }").WithLocation(12, 18), // (13,18): error CS0826: No best type found for implicitly-typed array - // var p5 = stackalloc[] { (1, null) }; + // var p4 = stackalloc[] { (1, null) }; Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { (1, null) }").WithLocation(13, 18), // (14,18): error CS0826: No best type found for implicitly-typed array - // var p6 = stackalloc[] { () => { } }; + // var p5 = stackalloc[] { () => { } }; Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { () => { } }").WithLocation(14, 18), // (15,18): error CS0826: No best type found for implicitly-typed array - // var p7 = stackalloc[] { new {} , new { i = 0 } }; - Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new {} , new { i = 0 } }").WithLocation(15, 18) + // var p6 = stackalloc[] { new {} , new { i = 0 } }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new {} , new { i = 0 } }").WithLocation(15, 18), + // (16,18): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // var p7 = stackalloc[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(16, 18), + // (17,33): error CS0103: The name '_' does not exist in the current context + // var p8 = stackalloc[] { _ }; + Diagnostic(ErrorCode.ERR_NameNotInContext, "_").WithArguments("_").WithLocation(17, 33) ); } [Fact] - public void BestTypeNumeric() + public void NoBestType_Span() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + struct A {} + struct B {} + + void Method(dynamic d, bool c) + { + var p0 = c ? default : stackalloc[] { new A(), new B() }; + var p1 = c ? default : stackalloc[] { }; + var p2 = c ? default : stackalloc[] { VoidMethod() }; + var p3 = c ? default : stackalloc[] { null }; + var p4 = c ? default : stackalloc[] { (1, null) }; + var p5 = c ? default : stackalloc[] { () => { } }; + var p6 = c ? default : stackalloc[] { new {} , new { i = 0 } }; + var p7 = c ? default : stackalloc[] { d }; + var p8 = c ? default : stackalloc[] { _ }; + } + + public void VoidMethod() {} +} +namespace System { + public struct ValueTuple { + public ValueTuple(T1 a, T2 b) { } + } +} +", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (9,32): error CS0826: No best type found for implicitly-typed array + // var p0 = c ? default : stackalloc[] { new A(), new B() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new A(), new B() }").WithLocation(9, 32), + // (10,32): error CS0826: No best type found for implicitly-typed array + // var p1 = c ? default : stackalloc[] { }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { }").WithLocation(10, 32), + // (11,32): error CS0826: No best type found for implicitly-typed array + // var p2 = c ? default : stackalloc[] { VoidMethod() }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { VoidMethod() }").WithLocation(11, 32), + // (12,32): error CS0826: No best type found for implicitly-typed array + // var p3 = c ? default : stackalloc[] { null }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { null }").WithLocation(12, 32), + // (13,32): error CS0826: No best type found for implicitly-typed array + // var p4 = c ? default : stackalloc[] { (1, null) }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { (1, null) }").WithLocation(13, 32), + // (14,32): error CS0826: No best type found for implicitly-typed array + // var p5 = c ? default : stackalloc[] { () => { } }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { () => { } }").WithLocation(14, 32), + // (15,32): error CS0826: No best type found for implicitly-typed array + // var p6 = c ? default : stackalloc[] { new {} , new { i = 0 } }; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[] { new {} , new { i = 0 } }").WithLocation(15, 32), + // (16,32): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('dynamic') + // var p7 = c ? default : stackalloc[] { d }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { d }").WithArguments("dynamic").WithLocation(16, 32), + // (17,47): error CS0103: The name '_' does not exist in the current context + // var p8 = c ? default : stackalloc[] { _ }; + Diagnostic(ErrorCode.ERR_NameNotInContext, "_").WithArguments("_").WithLocation(17, 47) + ); + } + + [Fact] + public void InitializeWithSelf_Pointer() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + void Method() + { + var obj1 = stackalloc int[1] { obj1 }; + var obj2 = stackalloc int[ ] { obj2 }; + var obj3 = stackalloc [ ] { obj3 }; + } +} +", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,40): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = stackalloc int[1] { obj1 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(6, 40), + // (7,40): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = stackalloc int[ ] { obj2 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(7, 40), + // (8,40): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = stackalloc [ ] { obj3 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(8, 40), + // (8,40): error CS0165: Use of unassigned local variable 'obj3' + // var obj3 = stackalloc [ ] { obj3 }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 40) + ); + } + + [Fact] + public void InitializeWithSelf_Span() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + void Method(bool c) + { + var obj1 = c ? default : stackalloc int[1] { obj1 }; + var obj2 = c ? default : stackalloc int[ ] { obj2 }; + var obj3 = c ? default : stackalloc [ ] { obj3 }; + } +} +", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,54): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = c ? default : stackalloc int[1] { obj1 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(6, 54), + // (7,54): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = c ? default : stackalloc int[ ] { obj2 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(7, 54), + // (8,54): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = c ? default : stackalloc [ ] { obj3 }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(8, 54), + // (8,54): error CS0165: Use of unassigned local variable 'obj3' + // var obj3 = c ? default : stackalloc [ ] { obj3 }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 54) + ); + } + + [Fact] + public void BestTypeNumeric_Pointer() { var comp = CreateCompilationWithMscorlibAndSpan(@" unsafe class Test @@ -97,25 +233,231 @@ public void Method1() } [Fact] - public void BadBestType() + public void BestTypeNumeric_Span() { var comp = CreateCompilationWithMscorlibAndSpan(@" unsafe class Test { - public void Method1() + public void Method1(bool c) + { + var obj1 = c ? default : stackalloc[] { 1, 1.2 }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); + Assert.Equal(1, variables.Count()); + + var obj1 = variables.ElementAt(0); + Assert.Equal("obj1", obj1.Identifier.Text); + + var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); + Assert.Equal(SpecialType.System_Double, ((NamedTypeSymbol)obj1Value.Type).TypeArgumentsNoUseSiteDiagnostics[0].SpecialType); + Assert.Equal(SpecialType.System_Double, ((NamedTypeSymbol)obj1Value.ConvertedType).TypeArgumentsNoUseSiteDiagnostics[0].SpecialType); + Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); + var declared = model.GetDeclaredSymbol(obj1.Initializer.Value); + Assert.Null(declared); + } + + [Fact] + public void BadBestType_Pointer() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + ref struct S {} + void Method1(S s) { var obj1 = stackalloc[] { """" }; var obj2 = stackalloc[] { new {} }; + var obj3 = stackalloc[] { s }; // OK } }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( - // (6,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('string') + // (7,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('string') // var obj1 = stackalloc[] { "" }; - Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(6, 20), - // (7,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('') + Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(7, 20), + // (8,20): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('') // var obj2 = stackalloc[] { new {} }; - Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(7, 20) + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(8, 20) + ); + } + + [Fact] + public void BadBestType_Span() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + ref struct S {} + void Method1(S s, bool c) + { + var obj1 = c ? default : stackalloc[] { """" }; + var obj2 = c ? default : stackalloc[] { new {} }; + var obj3 = c ? default : stackalloc[] { s }; // Should be an error, see https://github.com/dotnet/roslyn/issues/25086 + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (7,34): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('string') + // var obj1 = c ? default : stackalloc[] { "" }; + Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(7, 34), + // (8,34): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('') + // var obj2 = c ? default : stackalloc[] { new {} }; + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(8, 34) + ); + } + + [Fact] + public void TestFor_Pointer() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + static void Method1() + { + int i = 0; + for (var p = stackalloc int[3] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + static void Method2() + { + int i = 0; + for (var p = stackalloc int[ ] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + static void Method3() + { + int i = 0; + for (var p = stackalloc [ ] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + public static void Main() + { + Method1(); + Method2(); + Method3(); + } +}", TestOptions.UnsafeReleaseExe); + + CompileAndVerify(comp, expectedOutput: "123123123", verify: Verification.Fails); + } + + [Fact] + public void TestFor_Span() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +class Test +{ + static void Method1() + { + int i = 0; + for (Span p = stackalloc int[3] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + static void Method2() + { + int i = 0; + for (Span p = stackalloc int[ ] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + static void Method3() + { + int i = 0; + for (Span p = stackalloc [ ] { 1, 2, 3 }; i < 3; i++) + Console.Write(p[i]); + } + + public static void Main() + { + Method1(); + Method2(); + Method3(); + } +}", TestOptions.DebugExe); + + comp.VerifyDiagnostics(); + } + + [Fact] + public void TestAwait_Pointer() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System.Threading.Tasks; +unsafe class Test +{ + async void M() + { + var p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (7,32): error CS4004: Cannot await in an unsafe context + // var p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + Diagnostic(ErrorCode.ERR_AwaitInUnsafeContext, "await Task.FromResult(1)").WithLocation(7, 32), + // (7,60): error CS4004: Cannot await in an unsafe context + // var p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + Diagnostic(ErrorCode.ERR_AwaitInUnsafeContext, "await Task.FromResult(2)").WithLocation(7, 60) + ); + } + + [Fact] + public void TestAwait_Span() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +using System.Threading.Tasks; +class Test +{ + async void M() + { + Span p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (8,38): error CS0150: A constant value is expected + // Span p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + Diagnostic(ErrorCode.ERR_ConstantExpected, "await Task.FromResult(1)").WithLocation(8, 38), + // (8,9): error CS4012: Parameters or locals of type 'Span' cannot be declared in async methods or lambda expressions. + // Span p = stackalloc int[await Task.FromResult(1)] { await Task.FromResult(2) }; + Diagnostic(ErrorCode.ERR_BadSpecialByRefLocal, "Span").WithArguments("System.Span").WithLocation(8, 9) + ); + } + + [Fact] + public void TestSelfInSize() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +unsafe class Test +{ + void M() + { + var x = stackalloc int[x] { }; + } +}", TestOptions.UnsafeReleaseDll); + + comp.VerifyDiagnostics( + // (6,32): error CS0841: Cannot use local variable 'x' before it is declared + // var x = stackalloc int[x] { }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "x").WithArguments("x").WithLocation(6, 32), + // (6,32): error CS0165: Use of unassigned local variable 'x' + // var x = stackalloc int[x] { }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "x").WithArguments("x").WithLocation(6, 32) ); } From 2037b29bdcf4bfd432a4cefc044f16e89ede9c6e Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 2 Mar 2018 01:09:10 +0330 Subject: [PATCH 32/43] Delete stackalloc_initializers.md --- .../testplans/stackalloc_initializers.md | 61 ------------------- 1 file changed, 61 deletions(-) delete mode 100644 docs/compilers/CSharp/testplans/stackalloc_initializers.md diff --git a/docs/compilers/CSharp/testplans/stackalloc_initializers.md b/docs/compilers/CSharp/testplans/stackalloc_initializers.md deleted file mode 100644 index bed611dca773c..0000000000000 --- a/docs/compilers/CSharp/testplans/stackalloc_initializers.md +++ /dev/null @@ -1,61 +0,0 @@ - -Here is the rough testplan for the stackalloc initializers feature. - -Some tests may already exist or be part of other tests. -In such case we can just check them off. - - -## Evaluation semantics ## - -### correctness ### -- [ ] check that partial result is not visible. - if element throws and exception caught, whole thing is not reassigned to the new value. (try spans and ordinary stackallocs) -- [ ] if we refer to the elements of outer local in the initializers we still see the old values. -- [x] put `await` in the size or in the middle of element initializers -- [ ] initializer uses a local that is captured by a lambda expression. See that if lambda changes the value of the local, then we see the updated value when initializing - -### errors ## -- [x] invalid array shapes - multidimensional, nested, size mismatches, ... -- [x] bad conversions -- [x] bad local initialization contexts - using, fixed. -- [x] should it work in `for(int* x = stackalloc...)`, probably the same ways as before... ? -- [X] missing various span parts. -- [x] outer expression is not usable in elements (not assigned yet) -- [ ] in the array type inference case the outer expression cannot be used to determine the type - [ ] should be an error and should not be some kind of crash due to circular dependency -- [ ] make it to infer strange types. - - [x] dynamic - - [ ] ref-struct - - [x] void (use a void method to initialize an element) - - [X] lambda expression (formally typeless) - - [X] null - - [x] discard `_` - is not even a value. - -## API ## - -- [ ] GetSymbolInfo - - [X] on the whole node (checked for ordinary int* and Span) - - [ ] on size expression (including cases when implicit conversions happen) - - [ ] on element expressions (including cases when implicit conversions happen) - -- [ ] GetDeclaredSymbol - - [ ] the expression does not declare anything, try with whole node, elements, just in case.. - - - -## Manual IDE testing ## -The more detailed plan is here https://github.com/dotnet/roslyn/wiki/Manual-Testing although many items may end up N/A. -We will have to go through the list before merging. - -This may not be possible to be done with just public bits - I am not sure. Will check myself - - -Preliminary - the scenarios of concern are not many, since this is an expression and fairly simple too. - -- [ ] extract whole thing to a method (regardless whether that works for stackallocks, it should behave rationally, even if resulting code has errors due to escape rules, etc...) -- [ ] extract array size and elements to a method, especially if there are implicit conversions. (should just work if semantic APIs work correctly) -- [ ] general typing of the construct - - [ ] IDE should not autocorrect into something meaningless - - [ ] inside the initializer the dropdowns shoudl be as expected - i.e. variables in scope - - [ ] autocompletion of parens and braces - [] { } - - - From a1a7258f9147a8a31adca686a0ad9ebeb45c52de Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 2 Mar 2018 15:51:42 +0330 Subject: [PATCH 33/43] PR feedback --- .../Semantics/StackAllocInitializerTests.cs | 58 +++++++++++++++++-- 1 file changed, 54 insertions(+), 4 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 533eefb85e32d..e7e6daabfbb2c 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -144,12 +144,19 @@ public void InitializeWithSelf_Pointer() var comp = CreateCompilationWithMscorlibAndSpan(@" unsafe class Test { - void Method() + void Method1() { var obj1 = stackalloc int[1] { obj1 }; var obj2 = stackalloc int[ ] { obj2 }; var obj3 = stackalloc [ ] { obj3 }; } + + void Method2() + { + var obj1 = stackalloc int[2] { obj1[0] , obj1[1] }; + var obj2 = stackalloc int[ ] { obj2[0] , obj2[1] }; + var obj3 = stackalloc [ ] { obj3[0] , obj3[1] }; + } } ", TestOptions.UnsafeReleaseDll); @@ -165,7 +172,25 @@ void Method() Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(8, 40), // (8,40): error CS0165: Use of unassigned local variable 'obj3' // var obj3 = stackalloc [ ] { obj3 }; - Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 40) + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 40), + // (13,40): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = stackalloc int[2] { obj1[0] , obj1[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(13, 40), + // (13,50): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = stackalloc int[2] { obj1[0] , obj1[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(13, 50), + // (14,40): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = stackalloc int[ ] { obj2[0] , obj2[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(14, 40), + // (14,50): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = stackalloc int[ ] { obj2[0] , obj2[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(14, 50), + // (15,40): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = stackalloc [ ] { obj3[0] , obj3[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(15, 40), + // (15,50): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = stackalloc [ ] { obj3[0] , obj3[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(15, 50) ); } @@ -175,12 +200,19 @@ public void InitializeWithSelf_Span() var comp = CreateCompilationWithMscorlibAndSpan(@" unsafe class Test { - void Method(bool c) + void Method1(bool c) { var obj1 = c ? default : stackalloc int[1] { obj1 }; var obj2 = c ? default : stackalloc int[ ] { obj2 }; var obj3 = c ? default : stackalloc [ ] { obj3 }; } + + void Method2(bool c) + { + var obj1 = c ? default : stackalloc int[2] { obj1[0] , obj1[1] }; + var obj2 = c ? default : stackalloc int[ ] { obj2[0] , obj2[1] }; + var obj3 = c ? default : stackalloc [ ] { obj3[0] , obj3[1] }; + } } ", TestOptions.UnsafeReleaseDll); @@ -196,7 +228,25 @@ void Method(bool c) Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(8, 54), // (8,54): error CS0165: Use of unassigned local variable 'obj3' // var obj3 = c ? default : stackalloc [ ] { obj3 }; - Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 54) + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj3").WithArguments("obj3").WithLocation(8, 54), + // (13,54): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = c ? default : stackalloc int[2] { obj1[0] , obj1[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(13, 54), + // (13,64): error CS0841: Cannot use local variable 'obj1' before it is declared + // var obj1 = c ? default : stackalloc int[2] { obj1[0] , obj1[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj1").WithArguments("obj1").WithLocation(13, 64), + // (14,54): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = c ? default : stackalloc int[ ] { obj2[0] , obj2[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(14, 54), + // (14,64): error CS0841: Cannot use local variable 'obj2' before it is declared + // var obj2 = c ? default : stackalloc int[ ] { obj2[0] , obj2[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj2").WithArguments("obj2").WithLocation(14, 64), + // (15,54): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = c ? default : stackalloc [ ] { obj3[0] , obj3[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(15, 54), + // (15,64): error CS0841: Cannot use local variable 'obj3' before it is declared + // var obj3 = c ? default : stackalloc [ ] { obj3[0] , obj3[1] }; + Diagnostic(ErrorCode.ERR_VariableUsedBeforeDeclaration, "obj3").WithArguments("obj3").WithLocation(15, 64) ); } From 0fdddd9bd2c54e858e274423f206679d8c0ac807 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Mar 2018 17:54:30 +0330 Subject: [PATCH 34/43] Renamings --- ...itializerTests.cs => CodeGenStackAllocInitializerTests.cs} | 2 +- ...itializerTests.cs => StackAllocInitializerParsingTests.cs} | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) rename src/Compilers/CSharp/Test/Emit/CodeGen/{StackAllocInitializerTests.cs => CodeGenStackAllocInitializerTests.cs} (99%) rename src/Compilers/CSharp/Test/Syntax/Parsing/{StackAllocInitializerTests.cs => StackAllocInitializerParsingTests.cs} (98%) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs similarity index 99% rename from src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs rename to src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs index 47d9aaf93318b..ed44098a9d174 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs @@ -7,7 +7,7 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests { [CompilerTrait(CompilerFeature.StackAllocInitializer)] - public class StackAllocInitializerTests : CompilingTestBase + public class CodeGenStackAllocInitializerTests : CompilingTestBase { [Fact] public void TestElementThrow() diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs similarity index 98% rename from src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs rename to src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs index 22e71e0949d51..3e5e9363a01c3 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs @@ -8,9 +8,9 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests.Parsing { [CompilerTrait(CompilerFeature.StackAllocInitializer)] - public class StackAllocInitializerTests : ParsingTests + public class StackAllocInitializerParsingTests : ParsingTests { - public StackAllocInitializerTests(ITestOutputHelper output) : base(output) { } + public StackAllocInitializerParsingTests(ITestOutputHelper output) : base(output) { } [Fact] public void StackAllocInitializer_01() From 3eb28cca9ac1163ce525abe1c3bad020409fcaf7 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Mar 2018 18:22:42 +0330 Subject: [PATCH 35/43] Add symbol tests --- .../Semantics/StackAllocInitializerTests.cs | 206 ++++++++++++------ 1 file changed, 142 insertions(+), 64 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index e7e6daabfbb2c..c78a8da3c6bc6 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -250,70 +250,6 @@ void Method2(bool c) ); } - [Fact] - public void BestTypeNumeric_Pointer() - { - var comp = CreateCompilationWithMscorlibAndSpan(@" -unsafe class Test -{ - public void Method1() - { - var obj1 = stackalloc[] { 1, 1.2 }; - } -}", TestOptions.UnsafeReleaseDll); - - comp.VerifyDiagnostics( - ); - - var tree = comp.SyntaxTrees.Single(); - var model = comp.GetSemanticModel(tree); - - var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); - Assert.Equal(1, variables.Count()); - - var obj1 = variables.ElementAt(0); - Assert.Equal("obj1", obj1.Identifier.Text); - - var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); - Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.Type).PointedAtType.SpecialType); - Assert.Equal(SpecialType.System_Double, ((PointerTypeSymbol)obj1Value.ConvertedType).PointedAtType.SpecialType); - Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); - var declared = model.GetDeclaredSymbol(obj1.Initializer.Value); - Assert.Null(declared); - } - - [Fact] - public void BestTypeNumeric_Span() - { - var comp = CreateCompilationWithMscorlibAndSpan(@" -unsafe class Test -{ - public void Method1(bool c) - { - var obj1 = c ? default : stackalloc[] { 1, 1.2 }; - } -}", TestOptions.UnsafeReleaseDll); - - comp.VerifyDiagnostics( - ); - - var tree = comp.SyntaxTrees.Single(); - var model = comp.GetSemanticModel(tree); - - var variables = tree.GetCompilationUnitRoot().DescendantNodes().OfType(); - Assert.Equal(1, variables.Count()); - - var obj1 = variables.ElementAt(0); - Assert.Equal("obj1", obj1.Identifier.Text); - - var obj1Value = model.GetSemanticInfoSummary(obj1.Initializer.Value); - Assert.Equal(SpecialType.System_Double, ((NamedTypeSymbol)obj1Value.Type).TypeArgumentsNoUseSiteDiagnostics[0].SpecialType); - Assert.Equal(SpecialType.System_Double, ((NamedTypeSymbol)obj1Value.ConvertedType).TypeArgumentsNoUseSiteDiagnostics[0].SpecialType); - Assert.Equal(ConversionKind.Identity, obj1Value.ImplicitConversion.Kind); - var declared = model.GetDeclaredSymbol(obj1.Initializer.Value); - Assert.Null(declared); - } - [Fact] public void BadBestType_Pointer() { @@ -2014,5 +1950,147 @@ static int M(System.Action action) Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc [ ] { 1, 2, 3 }").WithLocation(96, 34) ); } + + [Fact] + public void StackAllocArrayCreationExpression_Symbols() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc double[2] { 1, 1.2 }; + Span obj2 = stackalloc double[2] { 1, 1.2 }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var expressions = tree.GetCompilationUnitRoot().DescendantNodes().OfType().ToArray(); + Assert.Equal(2, expressions.Length); + + var @stackalloc = expressions[0]; + var stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Double*", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Double*", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + var element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Int32", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + var element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Double", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element1Info.ImplicitConversion); + + var sizeInfo = model.GetSemanticInfoSummary(((ArrayTypeSyntax)@stackalloc.Type).RankSpecifiers[0].Sizes[0]); + Assert.Null(sizeInfo.Symbol); + Assert.Equal("System.Int32", sizeInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Int32", sizeInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, sizeInfo.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + + @stackalloc = expressions[1]; + stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Span", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Span", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Int32", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Double", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element1Info.ImplicitConversion); + + sizeInfo = model.GetSemanticInfoSummary(((ArrayTypeSyntax)@stackalloc.Type).RankSpecifiers[0].Sizes[0]); + Assert.Null(sizeInfo.Symbol); + Assert.Equal("System.Int32", sizeInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Int32", sizeInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, sizeInfo.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + } + + [Fact] + public void ImplicitStackAllocArrayCreationExpression_Symbols() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + var obj1 = stackalloc[] { 1, 1.2 }; + Span obj2 = stackalloc[] { 1, 1.2 }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics(); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var expressions = tree.GetCompilationUnitRoot().DescendantNodes().OfType().ToArray(); + Assert.Equal(2, expressions.Length); + + var @stackalloc = expressions[0]; + var stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Double*", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Double*", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + var element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Int32", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + var element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Double", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element1Info.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + + @stackalloc = expressions[1]; + stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Span", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Span", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Int32", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Double", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element1Info.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + } } } From 07ca9098ea9f50bc81ce7fb5ed742aceccc868b6 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Fri, 9 Mar 2018 23:33:14 +0330 Subject: [PATCH 36/43] Add capture test --- .../CodeGenStackAllocInitializerTests.cs | 53 +++++++++++++++++++ 1 file changed, 53 insertions(+) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs index ed44098a9d174..07d4302e0051b 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs @@ -9,6 +9,59 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests [CompilerTrait(CompilerFeature.StackAllocInitializer)] public class CodeGenStackAllocInitializerTests : CompilingTestBase { + [Fact] + public void TestLambdaCapture() + { + var text = @" +using System; +public class C +{ + unsafe public static void Main() + { + // captured by the lambda. + byte x = 1; + byte* b = stackalloc byte[] {((Func)(() => x++))(), x}; + Console.Write(b[1]); + } +} +"; + CompileAndVerify(text, + parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3), + options: TestOptions.UnsafeReleaseExe, + expectedOutput: "2", + verify: Verification.Fails).VerifyIL("C.Main", +@"{ + // Code size 55 (0x37) + .maxstack 4 + .locals init (C.<>c__DisplayClass0_0 V_0) //CS$<>8__locals0 + IL_0000: newobj ""C.<>c__DisplayClass0_0..ctor()"" + IL_0005: stloc.0 + IL_0006: ldloc.0 + IL_0007: ldc.i4.1 + IL_0008: stfld ""byte C.<>c__DisplayClass0_0.x"" + IL_000d: ldc.i4.2 + IL_000e: conv.u + IL_000f: localloc + IL_0011: dup + IL_0012: ldloc.0 + IL_0013: ldftn ""byte C.<>c__DisplayClass0_0.
b__0()"" + IL_0019: newobj ""System.Func..ctor(object, System.IntPtr)"" + IL_001e: callvirt ""byte System.Func.Invoke()"" + IL_0023: stind.i1 + IL_0024: dup + IL_0025: ldc.i4.1 + IL_0026: add + IL_0027: ldloc.0 + IL_0028: ldfld ""byte C.<>c__DisplayClass0_0.x"" + IL_002d: stind.i1 + IL_002e: ldc.i4.1 + IL_002f: add + IL_0030: ldind.u1 + IL_0031: call ""void System.Console.Write(int)"" + IL_0036: ret +}"); + } + [Fact] public void TestElementThrow() { From b9b337fc88fc4ffa721de9ea444a89662eff3c99 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Sat, 10 Mar 2018 00:37:39 +0330 Subject: [PATCH 37/43] Follow-up on merge from 'features/compiler' into stackalloc-init --- .../CodeGen/StackAllocInitializerTests.cs | 4 +-- .../Semantics/StackAllocInitializerTests.cs | 29 ++++++++++--------- 2 files changed, 18 insertions(+), 15 deletions(-) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs index 47d9aaf93318b..538c12d8b83bd 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/StackAllocInitializerTests.cs @@ -407,7 +407,7 @@ .maxstack 4 [Fact] public void TestSpan() { - var comp = CreateStandardCompilation(@" + var comp = CreateCompilation(@" using System; static class C { @@ -453,7 +453,7 @@ .locals init (int V_0) [Fact] public void TestReadOnlySpan() { - var comp = CreateStandardCompilation(@" + var comp = CreateCompilation(@" using System; static class C { diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index e7e6daabfbb2c..2315cb76e93aa 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -350,7 +350,7 @@ void Method1(S s, bool c) { var obj1 = c ? default : stackalloc[] { """" }; var obj2 = c ? default : stackalloc[] { new {} }; - var obj3 = c ? default : stackalloc[] { s }; // Should be an error, see https://github.com/dotnet/roslyn/issues/25086 + var obj3 = c ? default : stackalloc[] { s }; } }", TestOptions.UnsafeReleaseDll); @@ -360,7 +360,10 @@ void Method1(S s, bool c) Diagnostic(ErrorCode.ERR_ManagedAddr, @"stackalloc[] { """" }").WithArguments("string").WithLocation(7, 34), // (8,34): error CS0208: Cannot take the address of, get the size of, or declare a pointer to a managed type ('') // var obj2 = c ? default : stackalloc[] { new {} }; - Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(8, 34) + Diagnostic(ErrorCode.ERR_ManagedAddr, "stackalloc[] { new {} }").WithArguments("").WithLocation(8, 34), + // (9,34): error CS0306: The type 'Test.S' may not be used as a type argument + // var obj3 = c ? default : stackalloc[] { s }; + Diagnostic(ErrorCode.ERR_BadTypeArgument, "stackalloc[] { s }").WithArguments("Test.S").WithLocation(9, 34) ); } @@ -959,7 +962,7 @@ void Method3() [Fact] public void MissingSpanType() { - CreateStandardCompilation(@" + CreateCompilation(@" class Test { void M() @@ -993,7 +996,7 @@ void M() [Fact] public void MissingSpanConstructor() { - CreateStandardCompilation(@" + CreateCompilation(@" namespace System { ref struct Span @@ -1008,7 +1011,7 @@ void M() Span a3 = stackalloc [ ] { 1, 2, 3 }; } } -}").VerifyDiagnostics( +}").VerifyEmitDiagnostics( // (11,28): error CS0656: Missing compiler required member 'System.Span`1..ctor' // Span a1 = stackalloc int [3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_MissingPredefinedMember, "stackalloc int [3] { 1, 2, 3 }").WithArguments("System.Span`1", ".ctor").WithLocation(11, 28), @@ -1213,7 +1216,7 @@ void M() [Fact] public void StackAllocSyntaxProducesUnsafeErrorInSafeCode() { - CreateStandardCompilation(@" + CreateCompilation(@" class Test { void M() @@ -1336,7 +1339,7 @@ void M() } } "; - CreateStandardCompilation(test, options: new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary, allowUnsafe: true)).VerifyDiagnostics( + CreateCompilation(test, options: new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary, allowUnsafe: true)).VerifyDiagnostics( // (6,15): error CS0283: The type 'int*' cannot be declared const // const int* p1 = stackalloc int[1] { 1 }; Diagnostic(ErrorCode.ERR_BadConstType, "int*").WithArguments("int*").WithLocation(6, 15), @@ -1521,7 +1524,7 @@ static void Main() [Fact] public void StackAllocWithDynamic() { - CreateStandardCompilation(@" + CreateCompilation(@" class Program { static void Main() @@ -1583,7 +1586,7 @@ static void Main() [Fact] public void StackAllocAsArgument() { - CreateStandardCompilation(@" + CreateCompilation(@" class Program { static void N(object p) { } @@ -1610,7 +1613,7 @@ static void Main() [Fact] public void StackAllocInParenthesis() { - CreateStandardCompilation(@" + CreateCompilation(@" class Program { static void Main() @@ -1635,7 +1638,7 @@ static void Main() [Fact] public void StackAllocInNullConditionalOperator() { - CreateStandardCompilation(@" + CreateCompilation(@" class Program { static void Main() @@ -1793,7 +1796,7 @@ static int M(System.Action action) } } "; - CreateStandardCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + CreateCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( // (23,34): error CS0255: stackalloc may not be used in a catch or finally block // int* err11 = stackalloc int [3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(23, 34), @@ -1957,7 +1960,7 @@ static int M(System.Action action) } } "; - CreateStandardCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + CreateCompilation(text, options: TestOptions.UnsafeReleaseDll).VerifyDiagnostics( // (23,34): error CS0255: stackalloc may not be used in a catch or finally block // int* err11 = stackalloc int [3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_StackallocInCatchFinally, "stackalloc int [3] { 1, 2, 3 }").WithLocation(23, 34), From 29092d4fd116e985840be034546f2c996533530b Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Sat, 10 Mar 2018 00:12:49 +0330 Subject: [PATCH 38/43] Add IOperation tests --- .../CSharp/Portable/BoundTree/Expression.cs | 9 +- ...s_StackAllocArrayCreationAndInitializer.cs | 705 ++++++++++++++++++ 2 files changed, 712 insertions(+), 2 deletions(-) create mode 100644 src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs diff --git a/src/Compilers/CSharp/Portable/BoundTree/Expression.cs b/src/Compilers/CSharp/Portable/BoundTree/Expression.cs index 86db718813749..e41f90bec237e 100644 --- a/src/Compilers/CSharp/Portable/BoundTree/Expression.cs +++ b/src/Compilers/CSharp/Portable/BoundTree/Expression.cs @@ -106,12 +106,17 @@ internal partial class BoundFixedLocalCollectionInitializer internal partial class BoundStackAllocArrayCreation { - protected override ImmutableArray Children => ImmutableArray.Create(this.Count); + internal static ImmutableArray GetChildInitializers(BoundArrayInitialization arrayInitializer) + { + return arrayInitializer?.Initializers ?? ImmutableArray.Empty; + } + + protected override ImmutableArray Children => StaticCast.From(GetChildInitializers(this.InitializerOpt).Insert(0, this.Count)); } internal partial class BoundConvertedStackAllocExpression { - protected override ImmutableArray Children => ImmutableArray.Create(this.Count); + protected override ImmutableArray Children => StaticCast.From(GetChildInitializers(this.InitializerOpt).Insert(0, this.Count)); } internal partial class BoundDynamicObjectCreationExpression diff --git a/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs new file mode 100644 index 0000000000000..6f282e1fa6f81 --- /dev/null +++ b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs @@ -0,0 +1,705 @@ +// Copyright (c) Microsoft. All Rights Reserved. Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. + +using Microsoft.CodeAnalysis.CSharp.Syntax; +using Microsoft.CodeAnalysis.Test.Utilities; +using Roslyn.Test.Utilities; +using Xunit; + +namespace Microsoft.CodeAnalysis.CSharp.UnitTests +{ + public partial class IOperationTests : SemanticModelTestBase + { + [Fact] + public void SimpleStackAllocArrayCreation_PrimitiveType() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[1]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[1]') + Children(1): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid) (Syntax: '1') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[1]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[1]").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_UserDefinedType() + { + string source = @" +struct M { } + +class C +{ + public void F() + { + var a = /**/stackalloc M[1]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc M[1]') + Children(1): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid) (Syntax: '1') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[1]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[1]").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_ConstantDimension() + { + string source = @" +struct M { } + +class C +{ + public void F() + { + const int dimension = 1; + var a = /**/stackalloc M[dimension]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc M[dimension]') + Children(1): + ILocalReferenceOperation: dimension (OperationKind.LocalReference, Type: System.Int32, Constant: 1, IsInvalid) (Syntax: 'dimension') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(9,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[dimension]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[dimension]").WithLocation(9, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_NonConstantDimension() + { + string source = @" +struct M { } + +class C +{ + public void F(int dimension) + { + var a = /**/stackalloc M[dimension]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc M[dimension]') + Children(1): + IParameterReferenceOperation: dimension (OperationKind.ParameterReference, Type: System.Int32, IsInvalid) (Syntax: 'dimension') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[dimension]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[dimension]").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_DimensionWithImplicitConversion() + { + string source = @" +struct M { } + +class C +{ + public void F(char dimension) + { + var a = /**/stackalloc M[dimension]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc M[dimension]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, IsInvalid, IsImplicit) (Syntax: 'dimension') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: True, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IParameterReferenceOperation: dimension (OperationKind.ParameterReference, Type: System.Char, IsInvalid) (Syntax: 'dimension') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[dimension]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[dimension]").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_DimensionWithExplicitConversion() + { + string source = @" +struct M { } + +class C +{ + public void F(object dimension) + { + var a = /**/stackalloc M[(int)dimension]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc ... )dimension]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, IsInvalid) (Syntax: '(int)dimension') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: False, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IParameterReferenceOperation: dimension (OperationKind.ParameterReference, Type: System.Object, IsInvalid) (Syntax: 'dimension') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[(int)dimension]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[(int)dimension]").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializer_PrimitiveType() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[] { 42 }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[] { 42 }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid, IsImplicit) (Syntax: 'stackalloc int[] { 42 }') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 42, IsInvalid) (Syntax: '42') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[] { 42 }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[] { 42 }").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializer_PrimitiveTypeWithExplicitDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[1] { 42 }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[1] { 42 }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid) (Syntax: '1') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 42, IsInvalid) (Syntax: '42') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[1] { 42 }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[1] { 42 }").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializerErrorCase_PrimitiveTypeWithIncorrectExplicitDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[2] { 42 }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[2] { 42 }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 2, IsInvalid) (Syntax: '2') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 42, IsInvalid) (Syntax: '42') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0847: An array initializer of length '2' is expected + // var a = /**/stackalloc int[2] { 42 }/**/; + Diagnostic(ErrorCode.ERR_ArrayInitializerIncorrectLength, "stackalloc int[2] { 42 }").WithArguments("2").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializerErrorCase_PrimitiveTypeWithNonConstantExplicitDimension() + { + string source = @" +class C +{ + public void F(int dimension) + { + var a = /**/stackalloc int[dimension] { 42 }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc ... ion] { 42 }') + Children(2): + IParameterReferenceOperation: dimension (OperationKind.ParameterReference, Type: System.Int32, IsInvalid) (Syntax: 'dimension') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 42) (Syntax: '42') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,42): error CS0150: A constant value is expected + // var a = /**/stackalloc int[dimension] { 42 }/**/; + Diagnostic(ErrorCode.ERR_ConstantExpected, "dimension").WithLocation(6, 42) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializer_UserDefinedType() + { + string source = @" +struct M { } + +class C +{ + public void F() + { + var a = /**/stackalloc M[] { new M() }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc ... { new M() }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid, IsImplicit) (Syntax: 'stackalloc ... { new M() }') + IObjectCreationOperation (Constructor: M..ctor()) (OperationKind.ObjectCreation, Type: M, IsInvalid) (Syntax: 'new M()') + Arguments(0) + Initializer: + null +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc M[] { new M() }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc M[] { new M() }").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializer_ImplicitlyTyped() + { + string source = @" +struct M { } + +class C +{ + public void F() + { + var a = /**/stackalloc[] { new M() }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc[] { new M() }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid, IsImplicit) (Syntax: 'stackalloc[] { new M() }') + IObjectCreationOperation (Constructor: M..ctor()) (OperationKind.ObjectCreation, Type: M, IsInvalid) (Syntax: 'new M()') + Arguments(0) + Initializer: + null +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(8,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc[] { new M() }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc[] { new M() }").WithLocation(8, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializerErrorCase_ImplicitlyTypedWithoutInitializerAndDimension() + { + string source = @" +class C +{ + public void F(int dimension) + { + var x = /**/stackalloc[]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc[]/**/') + Children(1): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 0, IsInvalid, IsImplicit) (Syntax: 'stackalloc[]/**/') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,50): error CS1514: { expected + // var x = /**/stackalloc[]/**/; + Diagnostic(ErrorCode.ERR_LbraceExpected, ";").WithLocation(6, 50), + // file.cs(6,50): error CS1513: } expected + // var x = /**/stackalloc[]/**/; + Diagnostic(ErrorCode.ERR_RbraceExpected, ";").WithLocation(6, 50), + // file.cs(6,27): error CS0826: No best type found for implicitly-typed array + // var x = /**/stackalloc[]/**/; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[]/**/").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializerErrorCase_ImplicitlyTypedWithoutInitializer() + { + string source = @" +class C +{ + public void F(int dimension) + { + var x = /**/stackalloc[2]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc[2]/**/') + Children(1): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 0, IsInvalid, IsImplicit) (Syntax: 'stackalloc[2]/**/') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,38): error CS8373: "Invalid rank specifier: expected ']' + // var x = /**/stackalloc[2]/**/; + Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "2").WithLocation(6, 38), + // file.cs(6,51): error CS1514: { expected + // var x = /**/stackalloc[2]/**/; + Diagnostic(ErrorCode.ERR_LbraceExpected, ";").WithLocation(6, 51), + // file.cs(6,51): error CS1513: } expected + // var x = /**/stackalloc[2]/**/; + Diagnostic(ErrorCode.ERR_RbraceExpected, ";").WithLocation(6, 51), + // file.cs(6,27): error CS0826: No best type found for implicitly-typed array + // var x = /**/stackalloc[2]/**/; + Diagnostic(ErrorCode.ERR_ImplicitlyTypedArrayNoBestType, "stackalloc[2]/**/").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationWithInitializer_MultipleInitializersWithConversions() + { + string source = @" +class C +{ + public void F() + { + var a = 42; + var b = /**/stackalloc[] { 2, a, default }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc[ ... , default }') + Children(4): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 3, IsInvalid, IsImplicit) (Syntax: 'stackalloc[ ... , default }') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 2, IsInvalid) (Syntax: '2') + ILocalReferenceOperation: a (OperationKind.LocalReference, Type: System.Int32, IsInvalid) (Syntax: 'a') + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, Constant: 0, IsInvalid, IsImplicit) (Syntax: 'default') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: False, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IDefaultValueOperation (OperationKind.DefaultValue, Type: System.Int32, Constant: 0, IsInvalid) (Syntax: 'default') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(7,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var b = /**/stackalloc[] { 2, a, default }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc[] { 2, a, default }").WithLocation(7, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationErrorCase_MissingDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[]') + Children(1): + IInvalidOperation (OperationKind.Invalid, Type: ?, IsInvalid) (Syntax: '') + Children(0) +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,41): error CS1586: Array creation must have array size or array initializer + // var a = /**/stackalloc int[]/**/; + Diagnostic(ErrorCode.ERR_MissingArraySize, "[]").WithLocation(6, 41) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationErrorCase_InvalidInitializer() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[] { 1 }/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[] { 1 }') + Children(2): + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid, IsImplicit) (Syntax: 'stackalloc int[] { 1 }') + ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 1, IsInvalid) (Syntax: '1') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[] { 1 }/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[] { 1 }").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationErrorCase_MissingExplicitCast() + { + string source = @" +class C +{ + public void F(object b) + { + var a = /**/stackalloc int[b]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[b]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, IsInvalid, IsImplicit) (Syntax: 'b') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: False, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IParameterReferenceOperation: b (OperationKind.ParameterReference, Type: System.Object, IsInvalid) (Syntax: 'b') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,42): error CS0266: Cannot implicitly convert type 'object' to 'int'. An explicit conversion exists (are you missing a cast?) + // var a = /**/stackalloc int[b]/**/; + Diagnostic(ErrorCode.ERR_NoImplicitConvCast, "b").WithArguments("object", "int").WithLocation(6, 42), + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[b]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[b]").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreation_InvocationExpressionAsDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[M()]/**/; + } + + public int M() => 1; +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[M()]') + Children(1): + IInvocationOperation ( System.Int32 C.M()) (OperationKind.Invocation, Type: System.Int32, IsInvalid) (Syntax: 'M()') + Instance Receiver: + IInstanceReferenceOperation (OperationKind.InstanceReference, Type: C, IsInvalid, IsImplicit) (Syntax: 'M') + Arguments(0) +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[M()]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[M()]").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreation_InvocationExpressionWithConversionAsDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[(int)M()]/**/; + } + + public object M() => null; +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[(int)M()]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, IsInvalid) (Syntax: '(int)M()') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: False, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IInvocationOperation ( System.Object C.M()) (OperationKind.Invocation, Type: System.Object, IsInvalid) (Syntax: 'M()') + Instance Receiver: + IInstanceReferenceOperation (OperationKind.InstanceReference, Type: C, IsInvalid, IsImplicit) (Syntax: 'M') + Arguments(0) +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[(int)M()]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[(int)M()]").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationErrorCase_InvocationExpressionAsDimension() + { + string source = @" +class C +{ + public static void F() + { + var a = /**/stackalloc int[M()]/**/; + } + + public object M() => null; +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[M()]') + Children(1): + IInvocationOperation ( System.Object C.M()) (OperationKind.Invocation, Type: System.Object, IsInvalid) (Syntax: 'M()') + Instance Receiver: + IInstanceReferenceOperation (OperationKind.InstanceReference, Type: C, IsInvalid, IsImplicit) (Syntax: 'M') + Arguments(0) +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,42): error CS0120: An object reference is required for the non-static field, method, or property 'C.M()' + // var a = /**/stackalloc int[M()]/**/; + Diagnostic(ErrorCode.ERR_ObjectRequired, "M").WithArguments("C.M()").WithLocation(6, 42) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void StackAllocArrayCreationErrorCase_InvocationExpressionWithConversionAsDimension() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[(int)M()]/**/; + } + + public C M() => new C(); +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[(int)M()]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, IsInvalid) (Syntax: '(int)M()') + Conversion: CommonConversion (Exists: False, IsIdentity: False, IsNumeric: False, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + IInvocationOperation ( C C.M()) (OperationKind.Invocation, Type: C, IsInvalid) (Syntax: 'M()') + Instance Receiver: + IInstanceReferenceOperation (OperationKind.InstanceReference, Type: C, IsInvalid, IsImplicit) (Syntax: 'M') + Arguments(0) +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,42): error CS0030: Cannot convert type 'C' to 'int' + // var a = /**/stackalloc int[(int)M()]/**/; + Diagnostic(ErrorCode.ERR_NoExplicitConv, "(int)M()").WithArguments("C", "int").WithLocation(6, 42) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + + [Fact] + public void SimpleStackAllocArrayCreation_ConstantConversion() + { + string source = @" +class C +{ + public void F() + { + var a = /**/stackalloc int[0.0]/**/; + } +} +"; + string expectedOperationTree = @" +IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[0.0]') + Children(1): + IConversionOperation (TryCast: False, Unchecked) (OperationKind.Conversion, Type: System.Int32, Constant: 0, IsInvalid, IsImplicit) (Syntax: '0.0') + Conversion: CommonConversion (Exists: True, IsIdentity: False, IsNumeric: True, IsReference: False, IsUserDefined: False) (MethodSymbol: null) + Operand: + ILiteralOperation (OperationKind.Literal, Type: System.Double, Constant: 0, IsInvalid) (Syntax: '0.0') +"; + var expectedDiagnostics = new DiagnosticDescription[] { + // file.cs(6,42): error CS0266: Cannot implicitly convert type 'double' to 'int'. An explicit conversion exists (are you missing a cast?) + // var a = /**/stackalloc int[0.0]/**/; + Diagnostic(ErrorCode.ERR_NoImplicitConvCast, "0.0").WithArguments("double", "int").WithLocation(6, 42), + // file.cs(6,27): error CS0214: Pointers and fixed size buffers may only be used in an unsafe context + // var a = /**/stackalloc int[0.0]/**/; + Diagnostic(ErrorCode.ERR_UnsafeNeeded, "stackalloc int[0.0]").WithLocation(6, 27) + }; + + VerifyOperationTreeAndDiagnosticsForTest(source, expectedOperationTree, expectedDiagnostics); + } + } +} From 3d07cef351e59e2032baf41ebd822c1b921d98b5 Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Sat, 10 Mar 2018 20:15:57 +0330 Subject: [PATCH 39/43] Add symbol tests for error cases --- .../Semantics/StackAllocInitializerTests.cs | 162 ++++++++++++++++++ 1 file changed, 162 insertions(+) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index c78a8da3c6bc6..1acba9452c79c 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -2092,5 +2092,167 @@ public void Method1() Assert.Null(model.GetDeclaredSymbol(@stackalloc)); } + + [Fact] + public void StackAllocArrayCreationExpression_Symbols_ErrorCase() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + short* obj1 = stackalloc double[*obj1] { obj1[0], *obj1 }; + Span obj2 = stackalloc double[obj2.Length] { obj2[0], obj2.Length }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (7,41): error CS0150: A constant value is expected + // short* obj1 = stackalloc double[*obj1] { obj1[0], *obj1 }; + Diagnostic(ErrorCode.ERR_ConstantExpected, "*obj1").WithLocation(7, 41), + // (8,46): error CS0150: A constant value is expected + // Span obj2 = stackalloc double[obj2.Length] { obj2[0], obj2.Length }; + Diagnostic(ErrorCode.ERR_ConstantExpected, "obj2.Length").WithLocation(8, 46), + // (7,42): error CS0165: Use of unassigned local variable 'obj1' + // short* obj1 = stackalloc double[*obj1] { obj1[0], *obj1 }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj1").WithArguments("obj1").WithLocation(7, 42), + // (8,46): error CS0165: Use of unassigned local variable 'obj2' + // Span obj2 = stackalloc double[obj2.Length] { obj2[0], obj2.Length }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj2").WithArguments("obj2").WithLocation(8, 46) + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var expressions = tree.GetCompilationUnitRoot().DescendantNodes().OfType().ToArray(); + Assert.Equal(2, expressions.Length); + + var @stackalloc = expressions[0]; + var stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Null(stackallocInfo.Type); + Assert.Equal("System.Int16*", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.NoConversion, stackallocInfo.ImplicitConversion); + + var element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Int16", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + var element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Int16", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element1Info.ImplicitConversion); + + var sizeInfo = model.GetSemanticInfoSummary(((ArrayTypeSyntax)@stackalloc.Type).RankSpecifiers[0].Sizes[0]); + Assert.Null(sizeInfo.Symbol); + Assert.Equal("System.Int16", sizeInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Int32", sizeInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, sizeInfo.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + + @stackalloc = expressions[1]; + stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Null(stackallocInfo.Type); + Assert.Equal("System.Span", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.NoConversion, stackallocInfo.ImplicitConversion); + + element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Equal("ref System.Int16 System.Span.this[System.Int32 i] { get; }", element0Info.Symbol.ToTestDisplayString()); + Assert.Equal("System.Int16", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element0Info.ImplicitConversion); + + element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Equal("System.Int32 System.Span.Length { get; }", element1Info.Symbol.ToTestDisplayString()); + Assert.Equal("System.Int32", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element1Info.ImplicitConversion); + + sizeInfo = model.GetSemanticInfoSummary(((ArrayTypeSyntax)@stackalloc.Type).RankSpecifiers[0].Sizes[0]); + Assert.Equal("System.Int32 System.Span.Length { get; }", sizeInfo.Symbol.ToTestDisplayString()); + Assert.Equal("System.Int32", sizeInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Int32", sizeInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, sizeInfo.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + } + + [Fact] + public void ImplicitStackAllocArrayCreationExpression_Symbols_ErrorCase() + { + var comp = CreateCompilationWithMscorlibAndSpan(@" +using System; +unsafe class Test +{ + public void Method1() + { + double* obj1 = stackalloc[] { obj1[0], *obj1 }; + Span obj2 = stackalloc[] { obj2[0], obj2.Length }; + } +}", TestOptions.UnsafeReleaseDll).VerifyDiagnostics( + // (7,39): error CS0165: Use of unassigned local variable 'obj1' + // double* obj1 = stackalloc[] { obj1[0], *obj1 }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj1").WithArguments("obj1").WithLocation(7, 39), + // (8,44): error CS0165: Use of unassigned local variable 'obj2' + // Span obj2 = stackalloc[] { obj2[0], obj2.Length }; + Diagnostic(ErrorCode.ERR_UseDefViolation, "obj2").WithArguments("obj2").WithLocation(8, 44) + ); + + var tree = comp.SyntaxTrees.Single(); + var model = comp.GetSemanticModel(tree); + + var expressions = tree.GetCompilationUnitRoot().DescendantNodes().OfType().ToArray(); + Assert.Equal(2, expressions.Length); + + var @stackalloc = expressions[0]; + var stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Double*", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Double*", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + var element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Null(element0Info.Symbol); + Assert.Equal("System.Double", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element0Info.ImplicitConversion); + + var element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Null(element1Info.Symbol); + Assert.Equal("System.Double", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element1Info.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + + @stackalloc = expressions[1]; + stackallocInfo = model.GetSemanticInfoSummary(@stackalloc); + + Assert.Null(stackallocInfo.Symbol); + Assert.Equal("System.Span", stackallocInfo.Type.ToTestDisplayString()); + Assert.Equal("System.Span", stackallocInfo.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, stackallocInfo.ImplicitConversion); + + element0Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[0]); + Assert.Equal("ref System.Double System.Span.this[System.Int32 i] { get; }", element0Info.Symbol.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element0Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.Identity, element0Info.ImplicitConversion); + + element1Info = model.GetSemanticInfoSummary(@stackalloc.Initializer.Expressions[1]); + Assert.Equal("System.Int32 System.Span.Length { get; }", element1Info.Symbol.ToTestDisplayString()); + Assert.Equal("System.Int32", element1Info.Type.ToTestDisplayString()); + Assert.Equal("System.Double", element1Info.ConvertedType.ToTestDisplayString()); + Assert.Equal(Conversion.ImplicitNumeric, element1Info.ImplicitConversion); + + Assert.Null(model.GetDeclaredSymbol(@stackalloc)); + } } } From 6d45bafcf4c8ca7b4fd67a76f9635a90322220aa Mon Sep 17 00:00:00 2001 From: Alireza Habibi Date: Mon, 12 Mar 2018 21:19:13 +0330 Subject: [PATCH 40/43] Fix up IOperation test --- ...IOperationTests_StackAllocArrayCreationAndInitializer.cs | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs index 6f282e1fa6f81..05662383a3bcf 100644 --- a/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs +++ b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs @@ -622,10 +622,8 @@ public static void F() string expectedOperationTree = @" IOperation: (OperationKind.None, Type: null, IsInvalid) (Syntax: 'stackalloc int[M()]') Children(1): - IInvocationOperation ( System.Object C.M()) (OperationKind.Invocation, Type: System.Object, IsInvalid) (Syntax: 'M()') - Instance Receiver: - IInstanceReferenceOperation (OperationKind.InstanceReference, Type: C, IsInvalid, IsImplicit) (Syntax: 'M') - Arguments(0) + IInvalidOperation (OperationKind.Invalid, Type: System.Object, IsInvalid) (Syntax: 'M()') + Children(0) "; var expectedDiagnostics = new DiagnosticDescription[] { // file.cs(6,42): error CS0120: An object reference is required for the non-static field, method, or property 'C.M()' From bd0c63c9524256e6dc62bd64e98622ce925bf69d Mon Sep 17 00:00:00 2001 From: vsadov Date: Wed, 14 Mar 2018 15:24:55 -0700 Subject: [PATCH 41/43] Updated error IDs after merge --- .../Portable/xlf/CSharpResources.cs.xlf | 10 ++++ .../Portable/xlf/CSharpResources.de.xlf | 10 ++++ .../Portable/xlf/CSharpResources.es.xlf | 10 ++++ .../Portable/xlf/CSharpResources.fr.xlf | 10 ++++ .../Portable/xlf/CSharpResources.it.xlf | 10 ++++ .../Portable/xlf/CSharpResources.ja.xlf | 10 ++++ .../Portable/xlf/CSharpResources.ko.xlf | 10 ++++ .../Portable/xlf/CSharpResources.pl.xlf | 10 ++++ .../Portable/xlf/CSharpResources.pt-BR.xlf | 10 ++++ .../Portable/xlf/CSharpResources.ru.xlf | 10 ++++ .../Portable/xlf/CSharpResources.tr.xlf | 10 ++++ .../Portable/xlf/CSharpResources.zh-Hans.xlf | 10 ++++ .../Portable/xlf/CSharpResources.zh-Hant.xlf | 10 ++++ ...s_StackAllocArrayCreationAndInitializer.cs | 2 +- .../Semantics/StackAllocInitializerTests.cs | 2 +- .../StackAllocInitializerParsingTests.cs | 48 +++++++++---------- 16 files changed, 156 insertions(+), 26 deletions(-) diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf index 0926de5be9f16..d8618a74d43c3 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.cs.xlf @@ -8685,6 +8685,16 @@ Pokud chcete odstranit toto varování, můžete místo toho použít /reference Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf index 2f40d3d19e603..87e0591ddedda 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.de.xlf @@ -8685,6 +8685,16 @@ Um die Warnung zu beheben, können Sie stattdessen /reference verwenden (Einbett Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf index 304afb89cdb3e..7d3b6c988d3a7 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.es.xlf @@ -8685,6 +8685,16 @@ Para eliminar la advertencia puede usar /reference (establezca la propiedad Embe Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf index 9544dcf73c0b0..5336c20738a12 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.fr.xlf @@ -8685,6 +8685,16 @@ Pour supprimer l'avertissement, vous pouvez utiliser la commande /reference (dé Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf index 17c37461668f5..78a1746091c7c 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.it.xlf @@ -8685,6 +8685,16 @@ Per rimuovere l'avviso, è invece possibile usare /reference (impostare la propr Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf index ff3d6e1d08d99..717afa271cb6e 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ja.xlf @@ -8685,6 +8685,16 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf index 8eeb61bbab368..e6a0ba3001091 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ko.xlf @@ -8685,6 +8685,16 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf index c302307bafdf0..b4646b2bb4969 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pl.xlf @@ -8685,6 +8685,16 @@ Aby usunąć ostrzeżenie, możesz zamiast tego użyć opcji /reference (ustaw w Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf index edae634569241..8712191dd8d5e 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.pt-BR.xlf @@ -8685,6 +8685,16 @@ Para incorporar informações de tipo de interoperabilidade para os dois assembl Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf index c9e047494f210..7fcef595033e6 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.ru.xlf @@ -8685,6 +8685,16 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf index 145559423d915..b6c1c357093b2 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.tr.xlf @@ -8685,6 +8685,16 @@ Uyarıyı kaldırmak için, /reference kullanabilirsiniz (Birlikte Çalışma T Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf index 79967abcf5719..f7038ee837e3b 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hans.xlf @@ -8685,6 +8685,16 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf index 56b42b328a219..15d6badbb4a14 100644 --- a/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf +++ b/src/Compilers/CSharp/Portable/xlf/CSharpResources.zh-Hant.xlf @@ -8685,6 +8685,16 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ Type parameter '{1}' has the 'unmanaged' constraint so '{1}' cannot be used as a constraint for '{0}' + + stackalloc initializer + stackalloc initializer + + + + "Invalid rank specifier: expected ']' + "Invalid rank specifier: expected ']' + + \ No newline at end of file diff --git a/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs index 05662383a3bcf..17e1fa7a2e178 100644 --- a/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs +++ b/src/Compilers/CSharp/Test/Semantic/IOperation/IOperationTests_StackAllocArrayCreationAndInitializer.cs @@ -404,7 +404,7 @@ public void F(int dimension) ILiteralOperation (OperationKind.Literal, Type: System.Int32, Constant: 0, IsInvalid, IsImplicit) (Syntax: 'stackalloc[2]/**/') "; var expectedDiagnostics = new DiagnosticDescription[] { - // file.cs(6,38): error CS8373: "Invalid rank specifier: expected ']' + // file.cs(6,38): error CS8381: "Invalid rank specifier: expected ']' // var x = /**/stackalloc[2]/**/; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "2").WithLocation(6, 38), // file.cs(6,51): error CS1514: { expected diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs index 76cd9d429efb3..85d6cf9d587c5 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/StackAllocInitializerTests.cs @@ -584,7 +584,7 @@ public void Method1() }", TestOptions.UnsafeReleaseDll); comp.VerifyDiagnostics( - // (7,35): error CS8373: "Invalid rank specifier: expected ']' + // (7,35): error CS8381: "Invalid rank specifier: expected ']' // var obj2 = stackalloc [,] { 1 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(7, 35), // (6,31): error CS1575: A stackalloc expression requires [] after type diff --git a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs index 3e5e9363a01c3..78e4bc9ccf6e7 100644 --- a/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs +++ b/src/Compilers/CSharp/Test/Syntax/Parsing/StackAllocInitializerParsingTests.cs @@ -116,7 +116,7 @@ public void StackAllocInitializer_04() // (1,1): error CS8107: Feature 'stackalloc initializer' is not available in C# 7.0. Please use language version 7.3 or greater. // stackalloc[1] { 42 } Diagnostic(ErrorCode.ERR_FeatureNotAvailableInVersion7, "stackalloc").WithArguments("stackalloc initializer", "7.3").WithLocation(1, 1), - // (1,12): error CS8373: "Invalid rank specifier: expected ']' + // (1,12): error CS8381: "Invalid rank specifier: expected ']' // stackalloc[1] { 42 } Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "1").WithLocation(1, 12) ); @@ -150,7 +150,7 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28) ); @@ -168,10 +168,10 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29) ); @@ -189,10 +189,10 @@ void Goo() { "; ParseAndValidate(test, - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28) ); @@ -210,10 +210,10 @@ void Goo() { "; ParseAndValidate(test, - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3 { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3 { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), // (4,31): error CS1003: Syntax error, ']' expected @@ -234,7 +234,7 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3 { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), // (4,30): error CS1003: Syntax error, ']' expected @@ -255,10 +255,10 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3, { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3, { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), // (4,31): error CS1003: Syntax error, ']' expected @@ -279,13 +279,13 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), - // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // (4,30): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) ); @@ -303,13 +303,13 @@ void Goo() { "; ParseAndValidate(test, - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 29), - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), - // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // (4,30): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,3,] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) ); @@ -327,13 +327,13 @@ void Goo() { "; ParseAndValidate(test, - // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // (4,30): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 30), - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 28), - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29) ); @@ -351,16 +351,16 @@ void Goo() { "; ParseAndValidate(test, - // (4,28): error CS8373: "Invalid rank specifier: expected ']' + // (4,28): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 28), - // (4,31): error CS8373: "Invalid rank specifier: expected ']' + // (4,31): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, "3").WithLocation(4, 31), - // (4,29): error CS8373: "Invalid rank specifier: expected ']' + // (4,29): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 29), - // (4,30): error CS8373: "Invalid rank specifier: expected ']' + // (4,30): error CS8381: "Invalid rank specifier: expected ']' // var x = stackalloc[3,,3] { 1, 2, 3 }; Diagnostic(ErrorCode.ERR_InvalidStackAllocArray, ",").WithLocation(4, 30) ); From 96df47d31c33ac6203a6334161a2d86beced59d6 Mon Sep 17 00:00:00 2001 From: Jared Parsons Date: Thu, 15 Mar 2018 11:17:39 -0700 Subject: [PATCH 42/43] Remove props file reference There is no props file in this package to Import. Having the Import here is breaking our ability to publish in official builds. --- src/Tools/MicroBuild/PublishBlobAssets.proj | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Tools/MicroBuild/PublishBlobAssets.proj b/src/Tools/MicroBuild/PublishBlobAssets.proj index ba92e12f6c092..8ab9197bbbf01 100644 --- a/src/Tools/MicroBuild/PublishBlobAssets.proj +++ b/src/Tools/MicroBuild/PublishBlobAssets.proj @@ -10,7 +10,6 @@ This is for the internal orchestrated build scenarios and will likely never be run on a developer's machine. The official build definition builds this file directly. --> - From 9289f08030fe127386531748dd6eacc9587393ec Mon Sep 17 00:00:00 2001 From: Vladimir Sadov Date: Thu, 15 Mar 2018 19:10:35 -0700 Subject: [PATCH 43/43] Update test baselines after merge. --- .../CodeGenStackAllocInitializerTests.cs | 88 +++++++++---------- 1 file changed, 42 insertions(+), 46 deletions(-) diff --git a/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs index bdb0bdfdb1257..0e210d86efa63 100644 --- a/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs +++ b/src/Compilers/CSharp/Test/Emit/CodeGen/CodeGenStackAllocInitializerTests.cs @@ -483,24 +483,23 @@ public unsafe Span(void* p, int length) ", options: TestOptions.UnsafeReleaseExe, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3)); CompileAndVerify(comp, verify: Verification.Fails, expectedOutput: @"123") .VerifyIL("C.Main", -@"{ - // Code size 23 (0x17) +@" +{ + // Code size 21 (0x15) .maxstack 4 - .locals init (int V_0) IL_0000: ldc.i4.3 - IL_0001: stloc.0 - IL_0002: ldloc.0 - IL_0003: conv.u - IL_0004: localloc - IL_0006: dup - IL_0007: ldsflda "".__StaticArrayInitTypeSize=3 .7037807198C22A7D2B0807371D763779A84FDFCF"" - IL_000c: ldc.i4.3 - IL_000d: cpblk - IL_000f: ldloc.0 - IL_0010: newobj ""System.Span..ctor(void*, int)"" - IL_0015: pop - IL_0016: ret -}"); + IL_0001: conv.u + IL_0002: localloc + IL_0004: dup + IL_0005: ldsflda "".__StaticArrayInitTypeSize=3 .7037807198C22A7D2B0807371D763779A84FDFCF"" + IL_000a: ldc.i4.3 + IL_000b: cpblk + IL_000d: ldc.i4.3 + IL_000e: newobj ""System.Span..ctor(void*, int)"" + IL_0013: pop + IL_0014: ret +} +"); } [Fact] @@ -535,39 +534,36 @@ public readonly ref struct ReadOnlySpan ", options: TestOptions.UnsafeReleaseExe, parseOptions: TestOptions.Regular.WithLanguageVersion(LanguageVersion.CSharp7_3)); CompileAndVerify(comp, verify: Verification.Fails, expectedOutput: @"123") .VerifyIL("C.Main", -@"{ - // Code size 37 (0x25) +@" +{ + // Code size 34 (0x22) .maxstack 4 - .locals init (int V_0) - IL_0000: ldc.i4.3 - IL_0001: stloc.0 - IL_0002: ldloc.0 - IL_0003: conv.u - IL_0004: ldc.i4.4 - IL_0005: mul.ovf.un - IL_0006: localloc + IL_0000: ldc.i4.s 12 + IL_0002: conv.u + IL_0003: localloc + IL_0005: dup + IL_0006: ldc.i4.1 + IL_0007: stind.i4 IL_0008: dup - IL_0009: ldc.i4.1 - IL_000a: stind.i4 - IL_000b: dup - IL_000c: ldc.i4.4 - IL_000d: add + IL_0009: ldc.i4.4 + IL_000a: add + IL_000b: ldc.i4.2 + IL_000c: stind.i4 + IL_000d: dup IL_000e: ldc.i4.2 - IL_000f: stind.i4 - IL_0010: dup - IL_0011: ldc.i4.2 - IL_0012: conv.i - IL_0013: ldc.i4.4 - IL_0014: mul - IL_0015: add - IL_0016: ldc.i4.3 - IL_0017: stind.i4 - IL_0018: ldloc.0 - IL_0019: newobj ""System.Span..ctor(void*, int)"" - IL_001e: call ""System.ReadOnlySpan System.ReadOnlySpan.op_Implicit(System.Span)"" - IL_0023: pop - IL_0024: ret -}"); + IL_000f: conv.i + IL_0010: ldc.i4.4 + IL_0011: mul + IL_0012: add + IL_0013: ldc.i4.3 + IL_0014: stind.i4 + IL_0015: ldc.i4.3 + IL_0016: newobj ""System.Span..ctor(void*, int)"" + IL_001b: call ""System.ReadOnlySpan System.ReadOnlySpan.op_Implicit(System.Span)"" + IL_0020: pop + IL_0021: ret +} +"); } private static string GetSource(string pointerType) => $@"