Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Collection literals: address comments #68793

Merged
merged 7 commits into from
Jun 29, 2023
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 0 additions & 2 deletions src/Compilers/CSharp/Portable/Binder/Binder.ValueChecks.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3836,7 +3836,6 @@ internal uint GetValEscape(BoundExpression expr, uint scopeOfTheContainingExpres
return GetValEscape(switchExpr.SwitchArms.SelectAsArray(a => a.Value), scopeOfTheContainingExpression);

case BoundKind.CollectionLiteralExpression:
// PROTOTYPE: Revisit if spans may be optimized to avoid heap allocation.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We have a test plan #66418 item tracking this now.

return CallingMethodScope;

default:
Expand Down Expand Up @@ -4322,7 +4321,6 @@ internal bool CheckValEscape(SyntaxNode node, BoundExpression expr, uint escapeF
return true;

case BoundKind.CollectionLiteralExpression:
// PROTOTYPE: Revisit if spans may be optimized to avoid heap allocation.
return true;

default:
Expand Down
11 changes: 3 additions & 8 deletions src/Compilers/CSharp/Portable/Binder/Binder_Conversions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -464,7 +464,7 @@ private BoundCollectionLiteralExpression BindArrayOrSpanCollectionLiteral(
if (elements.Any(e => e is BoundCollectionLiteralSpreadElement))
{
// The array initializer includes at least one spread element, so we'll create an intermediate List<T> instance.
// PROTOTYPE: Avoid the intermediate list if the compile-time type of the spread element includes a length.
// https://github.com/dotnet/roslyn/issues/68785: Avoid intermediate List<T> if all spread elements have Length property.
_ = GetWellKnownTypeMember(WellKnownMember.System_Collections_Generic_List_T__ToArray, diagnostics, syntax: syntax);
var result = BindCollectionInitializerCollectionLiteral(
node,
Expand Down Expand Up @@ -531,7 +531,7 @@ private BoundCollectionLiteralExpression BindCollectionInitializerCollectionLite
if (targetType is NamedTypeSymbol namedType)
{
var analyzedArguments = AnalyzedArguments.GetInstance();
// PROTOTYPE: Should we use List<T>(int capacity) constructor when the size is known?
// https://github.com/dotnet/roslyn/issues/68785: Use ctor with `int capacity` when the size is known.
collectionCreation = BindClassCreationExpression(syntax, namedType.Name, syntax, namedType, analyzedArguments, diagnostics);
collectionCreation.WasCompilerGenerated = true;
analyzedArguments.Free();
Expand All @@ -548,18 +548,13 @@ private BoundCollectionLiteralExpression BindCollectionInitializerCollectionLite
}

var implicitReceiver = new BoundObjectOrCollectionValuePlaceholder(syntax, isNewInstance: true, targetType) { WasCompilerGenerated = true };
// PROTOTYPE: When generating a List<T>, should we use the well-known
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

PROTOTYPE

Added open issue to test plan #66418.

// member List<T>.Add() rather than relying on lookup?
var collectionInitializerAddMethodBinder = this.WithAdditionalFlags(BinderFlags.CollectionInitializerAddMethod);
var builder = ArrayBuilder<BoundExpression>.GetInstance(node.Elements.Length);
foreach (var element in node.Elements)
{
var result = element switch
{
BoundBadExpression => element,
// PROTOTYPE: Should spread elements support target type?
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

PROTOTYPE

Added open issue to test plan #66418.

// string[] a = [..b ? [null] : []];
// See CollectionLiteralTests.SpreadElement_05, _06.
BoundCollectionLiteralSpreadElement spreadElement => BindCollectionInitializerSpreadElementAddMethod(
(SpreadElementSyntax)spreadElement.Syntax,
spreadElement,
Expand Down Expand Up @@ -595,7 +590,7 @@ private BoundCollectionLiteralExpression BindListInterfaceCollectionLiteral(
TypeSymbol elementType,
BindingDiagnosticBag diagnostics)
{
// PROTOTYPE: Improve perf. For instance, emit [] as Array.Empty<T>() rather than a List<T>.
// https://github.com/dotnet/roslyn/issues/68785: Emit [] as Array.Empty<T>() rather than a List<T>.
var result = BindCollectionInitializerCollectionLiteral(
node,
CollectionLiteralTypeKind.ListInterface,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -571,7 +571,6 @@ public Conversion ClassifyStandardConversion(BoundExpression sourceExpression, T
return Conversion.NoConversion;
}

// PROTOTYPE: Ensure collection literal conversions are not considered standard implicit conversions.
private static bool IsStandardImplicitConversionFromExpression(ConversionKind kind)
{
if (IsStandardImplicitConversionFromType(kind))
Expand Down Expand Up @@ -1646,10 +1645,11 @@ static bool implementsIEnumerable(CSharpCompilation compilation, TypeSymbol targ
}

// This implementation differs from Binder.CollectionInitializerTypeImplementsIEnumerable().
// That method checks for an implicit conversion from IEnumerable to the collection type,
// but that would allow: Nullable<StructCollection> s = [];
// PROTOTYPE: Perhaps adjust the behavior of Binder.CollectionInitializerTypeImplementsIEnumerable()
// and use that method instead.
// That method checks for an implicit conversion from IEnumerable to the collection type, to
// match earlier implementation, even though it states that walking the implemented interfaces
// would be better. If we use CollectionInitializerTypeImplementsIEnumerable() here, we'd need
// to check for nullable to disallow: Nullable<StructCollection> s = [];
// Instead, we just walk the implemented interfaces.
var ienumerableType = compilation.GetSpecialType(SpecialType.System_Collections_IEnumerable);
return allInterfaces.Any(static (a, b) => areEqual(a, b), ienumerableType);
}
Expand Down
6 changes: 3 additions & 3 deletions src/Compilers/CSharp/Portable/Errors/ErrorCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2192,9 +2192,9 @@ internal enum ErrorCode
ERR_ConstantValueOfTypeExpected = 9135,
ERR_UnsupportedPrimaryConstructorParameterCapturingRefAny = 9136,

ERR_CollectionLiteralTargetTypeNotConstructible = 9500, // PROTOTYPE: Update error numbers.
ERR_ExpressionTreeContainsCollectionLiteral = 9501,
ERR_CollectionLiteralNoTargetType = 9503,
ERR_CollectionLiteralTargetTypeNotConstructible = 9174,
ERR_ExpressionTreeContainsCollectionLiteral = 9175,
ERR_CollectionLiteralNoTargetType = 9176,

#endregion

Expand Down
2 changes: 1 addition & 1 deletion src/Compilers/CSharp/Portable/Errors/MessageID.cs
Original file line number Diff line number Diff line change
Expand Up @@ -267,7 +267,7 @@ internal enum MessageID
IDS_FeatureUsingTypeAlias = MessageBase + 12834,

IDS_FeatureInstanceMemberInNameof = MessageBase + 12835,
IDS_FeatureCollectionLiterals = MessageBase + 12900, // PROTOTYPE: Update message number.
IDS_FeatureCollectionLiterals = MessageBase + 12837,
}

// Message IDs may refer to strings that need to be localized.
Expand Down
10 changes: 5 additions & 5 deletions src/Compilers/CSharp/Portable/FlowAnalysis/NullableWalker.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3473,12 +3473,12 @@ protected override void VisitStatement(BoundStatement statement)

public override BoundNode? VisitCollectionLiteralExpression(BoundCollectionLiteralExpression node)
{
// PROTOTYPE: Do we need to call inferInitialObjectState() to set the initial state of the instance?
// https://github.com/dotnet/roslyn/issues/68786: Use inferInitialObjectState() to set the initial
// state of the instance: see the call to InheritNullableStateOfTrackableStruct() in particular.
int containerSlot = GetOrCreatePlaceholderSlot(node);
bool delayCompletionForType = false; // PROTOTYPE: Should be true if the collection literal is target typed.
bool delayCompletionForType = false; // https://github.com/dotnet/roslyn/issues/68786: Should be true if the collection literal is target typed.

// PROTOTYPE: Test nullability of elements when the collection literal is target typed
// and the inferred target type has distinct element type nullability.
// https://github.com/dotnet/roslyn/issues/68786: Set nullability of elements from the inferred target type nullability.
foreach (var element in node.Elements)
{
switch (element)
Expand All @@ -3488,7 +3488,7 @@ protected override void VisitStatement(BoundStatement statement)
var completion = VisitCollectionElementInitializer(initializer, collectionType, delayCompletionForType);
if (completion is { })
{
// PROTOTYPE: Complete the analysis later.
// https://github.com/dotnet/roslyn/issues/68786: Complete the analysis later.
completion(containerSlot, collectionType);
}
break;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -61,17 +61,17 @@ private BoundExpression VisitArrayOrSpanCollectionLiteralExpression(BoundCollect
if (elements.Any(i => i is BoundCollectionLiteralSpreadElement))
{
// The array initializer includes at least one spread element, so we'll create an intermediate List<T> instance.
// PROTOTYPE: Avoid the intermediate list if the compile-time type of the spread element includes a length.
// PROTOTYPE: Use Enumerable.TryGetNonEnumeratedCount() at runtime for other cases.
// https://github.com/dotnet/roslyn/issues/68785: Avoid intermediate List<T> if all spread elements have Length property.
// https://github.com/dotnet/roslyn/issues/68785: Emit Enumerable.TryGetNonEnumeratedCount() and avoid intermediate List<T> at runtime.
var listType = _compilation.GetWellKnownType(WellKnownType.System_Collections_Generic_List_T).Construct(elementType);
var listToArray = ((MethodSymbol)_compilation.GetWellKnownTypeMember(WellKnownMember.System_Collections_Generic_List_T__ToArray)!).AsMember(listType);
var list = VisitCollectionInitializerCollectionLiteralExpression(node);
array = _factory.Call(list, listToArray); // PROTOTYPE: Improve perf. For instance, avoid copying to a new array.
array = _factory.Call(list, listToArray);
}
else
{
int arrayLength = elements.Length;
// PROTOTYPE: Should [] be emitted as Array.Empty<T>()?
// https://github.com/dotnet/roslyn/issues/68785: Emit [] as Array.Empty<T>() rather than a List<T>.
var initialization = (arrayLength == 0)
? null
: new BoundArrayInitialization(
Expand Down Expand Up @@ -149,7 +149,7 @@ private BoundExpression VisitListInterfaceCollectionLiteralExpression(BoundColle
Debug.Assert(!_inExpressionLambda);
Debug.Assert(node.Type is { });

// PROTOTYPE: Improve perf. For instance, emit [] as Array.Empty<T>() rather than a List<T>.
// https://github.com/dotnet/roslyn/issues/68785: Emit [] as Array.Empty<T>() rather than a List<T>.
var list = VisitCollectionInitializerCollectionLiteralExpression(node);
return _factory.Convert(node.Type, list);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -527,7 +527,7 @@ public void F()
";
var expectedDiagnostics = new DiagnosticDescription[]
{
// (6,27): error CS9503: There is no target type for the collection literal.
// (6,27): error CS9176: There is no target type for the collection literal.
// var a = /*<bind>*/[0]/*</bind>*/;
Diagnostic(ErrorCode.ERR_CollectionLiteralNoTargetType, "[0]").WithLocation(6, 27)
};
Expand Down
Loading