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

Overload Resolution Priority #74190

Merged
merged 26 commits into from
Jul 10, 2024
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
9b97133
Initial implementation of method symbols. Many more tests need to be …
333fred May 4, 2024
c5dfbe8
Add application validation and more testing
333fred Jun 21, 2024
76716e0
Add support for constructors, add some tests around cycles.
333fred Jun 21, 2024
409da23
Increase test coverage and mitigate a binding cycle with constructors.
333fred Jun 24, 2024
3123da0
Add support for indexers
333fred Jun 25, 2024
aff5004
Add several new tests.
333fred Jun 25, 2024
b50bde6
Add many more tests, error when putting overload resolution priority …
333fred Jun 26, 2024
5d17914
Adjust test baseline after rebase
333fred Jun 27, 2024
524a0e2
Update language feature status.
333fred Jun 27, 2024
014f1b3
Broaden early out.
333fred Jun 27, 2024
349b1e2
More accessor tests, support erroring on event accessors.
333fred Jun 27, 2024
d57d1d0
Add more langversion tests
333fred Jun 28, 2024
a83752e
PR feedback, support explicit implementations.
333fred Jul 1, 2024
08382a8
Typo
333fred Jul 1, 2024
b2904ac
Missing attributes.
333fred Jul 1, 2024
196eea0
Respond to PR feedback.
333fred Jul 3, 2024
c0c1730
PR feedback. Simplify CanHaveOverloadResolutionPriority significantly…
333fred Jul 8, 2024
e833d16
More tests from test plan review
333fred Jul 9, 2024
730aea7
Merge remote-tracking branch 'upstream/main' into feature/overload-re…
333fred Jul 9, 2024
f3c6c98
Update LangVersion test after merge with main, address final prototypes.
333fred Jul 9, 2024
336236d
Latest round of PR feedback and additional testing.
333fred Jul 9, 2024
1cba1e8
Final pieces of feedback from Julien
333fred Jul 9, 2024
ae1569e
Add OverloadResolutionPriorityAttribute to the compiler test plan.
333fred Jul 9, 2024
d46b524
Remove asserts that can be hit in rare bad metadata cases.
333fred Jul 9, 2024
50112d3
Beef up a cycle test and address a race condition. We were binding ob…
333fred Jul 10, 2024
a0d6cb4
PR feedback from Chuck
333fred Jul 10, 2024
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: 1 addition & 1 deletion .vscode/settings.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,5 +35,5 @@
"azure-pipelines.customSchemaFile": ".vscode/dnceng-schema.json",
"dotnet.defaultSolution": "Roslyn.sln",
"dotnet.completion.showCompletionItemsFromUnimportedNamespaces": true,
"dotnet.testWindow.disableAutoDiscovery": true
"dotnet.testWindow.disableAutoDiscovery": false
333fred marked this conversation as resolved.
Show resolved Hide resolved
}
2 changes: 1 addition & 1 deletion docs/Language Feature Status.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ efforts behind them.
| [Params-collections](https://github.com/dotnet/csharplang/issues/7700) | main | [Merged to 17.10p3](https://github.com/dotnet/roslyn/issues/71137) | [AlekseyTs](https://github.com/AlekseyTs) | [RikkiGibson](https://github.com/RikkiGibson), [333fred](https://github.com/333fred) | [akhera99](https://github.com/akhera99) (needs IDE fixer) | [MadsTorgersen](https://github.com/MadsTorgersen), [AlekseyTs](https://github.com/AlekseyTs) |
| [Ref/unsafe in iterators/async](https://github.com/dotnet/csharplang/blob/main/proposals/ref-unsafe-in-iterators-async.md) | [RefInAsync](https://github.com/dotnet/roslyn/tree/features/RefInAsync) | [Merged into 17.11p2](https://github.com/dotnet/roslyn/issues/72662) | [jjonescz](https://github.com/jjonescz) | [AlekseyTs](https://github.com/AlekseyTs), [cston](https://github.com/cston) | (no IDE impact) | |
| [Ref Struct Interfaces](https://github.com/dotnet/csharplang/issues/7608) | [RefStructInterfaces](https://github.com/dotnet/roslyn/tree/features/RefStructInterfaces) | [Merged into 17.11p2](https://github.com/dotnet/roslyn/issues/72124) | [AlekseyTs](https://github.com/AlekseyTs) | [cston](https://github.com/cston), [jjonescz](https://github.com/jjonescz) | [ToddGrun](https://github.com/ToddGrun) | [agocke](https://github.com/agocke), [jaredpar](https://github.com/jaredpar) |
| [Overload Resolution Priority](https://github.com/dotnet/csharplang/issues/7706) | PR not yet available | [In Progress](https://github.com/dotnet/roslyn/issues/74131) | [333fred](https://github.com/333fred) | [jcouv](https://github.com/jcouv), [cston](https://github.com/cston) | Not yet assigned | [333fred](https://github.com/333fred) |
| [Overload Resolution Priority](https://github.com/dotnet/csharplang/issues/7706) | main | [Merged to 17.12p1](https://github.com/dotnet/roslyn/issues/74131) | [333fred](https://github.com/333fred) | [jcouv](https://github.com/jcouv), [cston](https://github.com/cston) | Not yet assigned | [333fred](https://github.com/333fred) |
| [Partial properties](https://github.com/dotnet/csharplang/issues/6420) | [partial-properties](https://github.com/dotnet/roslyn/tree/features/partial-properties) | [Merged into 17.11p3](https://github.com/dotnet/roslyn/issues/73090) | [RikkiGibson](https://github.com/RikkiGibson) | [jcouv](https://github.com/jcouv), [333fred](https://github.com/333fred) | [Cosifne](https://github.com/Cosifne) | [333fred](https://github.com/333fred), [RikkiGibson](https://github.com/RikkiGibson) |

# C# 12.0
Expand Down
6 changes: 3 additions & 3 deletions src/Compilers/CSharp/Portable/Binder/Binder_Expressions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6543,7 +6543,7 @@ protected BoundExpression BindClassCreationExpression(
CompoundUseSiteInfo<AssemblySymbol> useSiteInfo = GetNewCompoundUseSiteInfo(diagnostics);
OverloadResolutionResult<MethodSymbol> overloadResolutionResult = OverloadResolutionResult<MethodSymbol>.GetInstance();
ImmutableArray<MethodSymbol> accessibleConstructors = GetAccessibleConstructorsForOverloadResolution(type, ref useSiteInfo);
this.OverloadResolution.ObjectCreationOverloadResolution(accessibleConstructors, analyzedArguments, overloadResolutionResult, dynamicResolution: true, ref useSiteInfo);
this.OverloadResolution.ObjectCreationOverloadResolution(accessibleConstructors, analyzedArguments, overloadResolutionResult, dynamicResolution: true, isEarlyAttributeBinding: IsEarlyAttributeBinder, ref useSiteInfo);

if (overloadResolutionResult.HasAnyApplicableMember)
{
Expand Down Expand Up @@ -7014,7 +7014,7 @@ internal bool TryPerformConstructorOverloadResolution(
if (candidateConstructors.Any())
{
// We have at least one accessible candidate constructor, perform overload resolution with accessible candidateConstructors.
this.OverloadResolution.ObjectCreationOverloadResolution(candidateConstructors, analyzedArguments, result, dynamicResolution: false, ref useSiteInfo);
this.OverloadResolution.ObjectCreationOverloadResolution(candidateConstructors, analyzedArguments, result, dynamicResolution: false, isEarlyAttributeBinding: IsEarlyAttributeBinder, ref useSiteInfo);

if (result.Succeeded)
{
Expand All @@ -7028,7 +7028,7 @@ internal bool TryPerformConstructorOverloadResolution(
// We might have a best match constructor which is inaccessible.
// Try overload resolution with all instance constructors to generate correct diagnostics and semantic info for this case.
OverloadResolutionResult<MethodSymbol> inaccessibleResult = OverloadResolutionResult<MethodSymbol>.GetInstance();
this.OverloadResolution.ObjectCreationOverloadResolution(allInstanceConstructors, analyzedArguments, inaccessibleResult, dynamicResolution: false, ref useSiteInfo);
this.OverloadResolution.ObjectCreationOverloadResolution(allInstanceConstructors, analyzedArguments, inaccessibleResult, dynamicResolution: false, isEarlyAttributeBinding: IsEarlyAttributeBinder, ref useSiteInfo);

if (inaccessibleResult.Succeeded)
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -98,20 +98,20 @@ private static bool SingleValidResult<TMember>(ArrayBuilder<MemberResolutionResu

// Perform overload resolution on the given method group, with the given arguments and
// names. The names can be null if no names were supplied to any arguments.
public void ObjectCreationOverloadResolution(ImmutableArray<MethodSymbol> constructors, AnalyzedArguments arguments, OverloadResolutionResult<MethodSymbol> result, bool dynamicResolution, ref CompoundUseSiteInfo<AssemblySymbol> useSiteInfo)
public void ObjectCreationOverloadResolution(ImmutableArray<MethodSymbol> constructors, AnalyzedArguments arguments, OverloadResolutionResult<MethodSymbol> result, bool dynamicResolution, bool isEarlyAttributeBinding, ref CompoundUseSiteInfo<AssemblySymbol> useSiteInfo)
{
Debug.Assert(!dynamicResolution || arguments.HasDynamicArgument);

var results = result.ResultsBuilder;

// First, attempt overload resolution not getting complete results.
PerformObjectCreationOverloadResolution(results, constructors, arguments, completeResults: false, dynamicResolution: dynamicResolution, ref useSiteInfo);
PerformObjectCreationOverloadResolution(results, constructors, arguments, completeResults: false, dynamicResolution: dynamicResolution, isEarlyAttributeBinding, ref useSiteInfo);

if (!OverloadResolutionResultIsValid(results, arguments.HasDynamicArgument))
{
// We didn't get a single good result. Get full results of overload resolution and return those.
result.Clear();
PerformObjectCreationOverloadResolution(results, constructors, arguments, completeResults: true, dynamicResolution: dynamicResolution, ref useSiteInfo);
PerformObjectCreationOverloadResolution(results, constructors, arguments, completeResults: true, dynamicResolution: dynamicResolution, isEarlyAttributeBinding, ref useSiteInfo);
}
}

Expand Down Expand Up @@ -361,6 +361,8 @@ private void PerformMemberOverloadResolution<TMember>(

if ((options & Options.DynamicResolution) == 0)
333fred marked this conversation as resolved.
Show resolved Hide resolved
{
RemoveLowerPriorityMembers(results);
jcouv marked this conversation as resolved.
Show resolved Hide resolved

// SPEC: The best method of the set of candidate methods is identified. If a single best method cannot be identified,
// SPEC: the method invocation is ambiguous, and a binding-time error occurs.

Expand Down Expand Up @@ -1603,6 +1605,7 @@ private void PerformObjectCreationOverloadResolution(
AnalyzedArguments arguments,
bool completeResults,
bool dynamicResolution,
bool isEarlyAttributeBinding,
ref CompoundUseSiteInfo<AssemblySymbol> useSiteInfo)
{
// SPEC: The instance constructor to invoke is determined using the overload resolution
Expand All @@ -1620,6 +1623,16 @@ private void PerformObjectCreationOverloadResolution(

if (!dynamicResolution)
333fred marked this conversation as resolved.
Show resolved Hide resolved
{
if (!isEarlyAttributeBinding)
{
// If we're still decoding early attributes, we can get into a cycle here where we attempt to decode early attributes,
// which causes overload resolution, which causes us to attempt to decode early attributes, etc. Concretely, this means
// that OverloadResolutionPriorityAttribute won't affect early bound attributes, so you can't use OverloadResolutionPriorityAttribute
// to adjust what constructor of OverloadResolutionPriorityAttribute is chosen. See `CycleOnOverloadResolutionPriorityConstructor_02` for
// an example.
RemoveLowerPriorityMembers(results);
}

// The best method of the set of candidate methods is identified. If a single best
// method cannot be identified, the method invocation is ambiguous, and a binding-time
// error occurs.
Expand Down Expand Up @@ -1698,6 +1711,78 @@ private int GetTheBestCandidateIndex<TMember>(ArrayBuilder<MemberResolutionResul
return currentBestIndex;
}

private static void RemoveLowerPriorityMembers<TMember>(ArrayBuilder<MemberResolutionResult<TMember>> results)
333fred marked this conversation as resolved.
Show resolved Hide resolved
where TMember : Symbol
{
// - Then, the reduced set of candidate members is grouped by declaring type. Within each group:
// - Candidate function members are ordered by *overload_resolution_priority*.
// - All members that have a lower *overload_resolution_priority* than the highest found within its declaring type group are removed.
// - The reduced groups are then recombined into the final set of applicable candidate function members.

switch (results)
{
// Can't prune anything unless there's at least 2 candidates
case []:
case [_]:
// We only look at methods and indexers, so if this isn't one of those scenarios, we don't need to do anything.
333fred marked this conversation as resolved.
Show resolved Hide resolved
case [{ Member: not (MethodSymbol or PropertySymbol { IsIndexer: true }) }, _, ..]:
jcouv marked this conversation as resolved.
Show resolved Hide resolved
return;
}

// Attempt to avoid any allocations by starting with a quick pass through all results and seeing if any have non-default priority. If so, we'll do the full sort and filter.
if (results.All(r => r.LeastOverriddenMember.GetOverloadResolutionPriority() == 0))
333fred marked this conversation as resolved.
Show resolved Hide resolved
{
// All default, nothing to do
return;
}

bool removedMembers = false;
var resultsByContainingType = PooledDictionary<NamedTypeSymbol, OneOrMany<MemberResolutionResult<TMember>>>.GetInstance();

foreach (var result in results)
{
var containingType = result.LeastOverriddenMember.ContainingType;
if (resultsByContainingType.TryGetValue(containingType, out var previousResults))
{
var previousOverloadResolutionPriority = previousResults.First().LeastOverriddenMember.GetOverloadResolutionPriority();
var currentOverloadResolutionPriority = result.LeastOverriddenMember.GetOverloadResolutionPriority();

if (currentOverloadResolutionPriority > previousOverloadResolutionPriority)
{
removedMembers = true;
resultsByContainingType[containingType] = OneOrMany.Create(result);
}
else if (currentOverloadResolutionPriority == previousOverloadResolutionPriority)
{
resultsByContainingType[containingType] = previousResults.Add(result);
}
else
{
removedMembers = true;
Debug.Assert(previousResults.All(r => r.LeastOverriddenMember.GetOverloadResolutionPriority() > currentOverloadResolutionPriority));
333fred marked this conversation as resolved.
Show resolved Hide resolved
}
}
else
{
resultsByContainingType.Add(containingType, OneOrMany.Create(result));
}
}

if (!removedMembers)
{
// No changes, so we can just return
resultsByContainingType.Free();
return;
}

results.Clear();
foreach (var (_, resultsForType) in resultsByContainingType)
{
results.AddRange(resultsForType);
}
resultsByContainingType.Free();
}

private void RemoveWorseMembers<TMember>(ArrayBuilder<MemberResolutionResult<TMember>> results, AnalyzedArguments arguments, ref CompoundUseSiteInfo<AssemblySymbol> useSiteInfo)
where TMember : Symbol
{
Expand Down
13 changes: 13 additions & 0 deletions src/Compilers/CSharp/Portable/CSharpResources.resx
Original file line number Diff line number Diff line change
Expand Up @@ -7974,6 +7974,19 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ
<data name="IDS_FeatureAllowsRefStructConstraint" xml:space="preserve">
<value>allows ref struct constraint</value>
</data>
<!-- PROTOTYPE: Condense -->
<data name="ERR_CannotApplyOverloadResolutionPriorityToOverride" xml:space="preserve">
333fred marked this conversation as resolved.
Show resolved Hide resolved
<value>Cannot put 'OverloadResolutionPriorityAttribute' on an overriding member.</value>
333fred marked this conversation as resolved.
Show resolved Hide resolved
</data>
<data name="ERR_CannotApplyOverloadResolutionPriorityToNonIndexer" xml:space="preserve">
<value>Cannot put 'OverloadResolutionPriorityAttribute' on a property that is not an indexer.</value>
</data>
<data name="ERR_CannotApplyOverloadResolutionPriorityToAccessor" xml:space="preserve">
<value>Cannot put 'OverloadResolutionPriorityAttribute' on an accessor method.</value>
</data>
<data name="IDS_OverloadResolutionPriority" xml:space="preserve">
<value>overload resolution priority</value>
</data>
<data name="ERR_InlineArrayAttributeOnRecord" xml:space="preserve">
<value>Attribute 'System.Runtime.CompilerServices.InlineArray' cannot be applied to a record struct.</value>
</data>
Expand Down
5 changes: 5 additions & 0 deletions src/Compilers/CSharp/Portable/Errors/ErrorCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2346,5 +2346,10 @@ internal enum ErrorCode
// Note: you will need to do the following after adding warnings:
// 1) Re-generate compiler code (eng\generate-compiler-code.cmd).
// 2) Update ErrorFacts.IsBuildOnlyDiagnostic (src/Compilers/CSharp/Portable/Errors/ErrorFacts.cs)

// PROTOTYPE: Condense
333fred marked this conversation as resolved.
Show resolved Hide resolved
ERR_CannotApplyOverloadResolutionPriorityToOverride = 9500,
ERR_CannotApplyOverloadResolutionPriorityToNonIndexer = 9501,
ERR_CannotApplyOverloadResolutionPriorityToAccessor = 9502,
}
}
3 changes: 3 additions & 0 deletions src/Compilers/CSharp/Portable/Errors/ErrorFacts.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2451,6 +2451,9 @@ or ErrorCode.WRN_PartialPropertySignatureDifference
or ErrorCode.ERR_PartialPropertyRequiredDifference
or ErrorCode.INF_IdentifierConflictWithContextualKeyword
or ErrorCode.ERR_InlineArrayAttributeOnRecord
or ErrorCode.ERR_CannotApplyOverloadResolutionPriorityToOverride
or ErrorCode.ERR_CannotApplyOverloadResolutionPriorityToNonIndexer
or ErrorCode.ERR_CannotApplyOverloadResolutionPriorityToAccessor
=> false,
};
#pragma warning restore CS8524 // The switch expression does not handle some values of its input type (it is not exhaustive) involving an unnamed enum value.
Expand Down
4 changes: 4 additions & 0 deletions src/Compilers/CSharp/Portable/Errors/MessageID.cs
Original file line number Diff line number Diff line change
Expand Up @@ -290,6 +290,9 @@ internal enum MessageID
IDS_FeaturePartialProperties = MessageBase + 12845,
IDS_FeatureFieldAndValueKeywords = MessageBase + 12846,

// PROTOTYPE: condense
IDS_OverloadResolutionPriority = MessageBase + 12900,

IDS_FeatureAllowsRefStructConstraint = MessageBase + 12847,
}

Expand Down Expand Up @@ -480,6 +483,7 @@ internal static LanguageVersion RequiredVersion(this MessageID feature)
case MessageID.IDS_FeatureAllowsRefStructConstraint:
case MessageID.IDS_FeaturePartialProperties:
case MessageID.IDS_FeatureFieldAndValueKeywords:
case MessageID.IDS_OverloadResolutionPriority:
return LanguageVersion.Preview;

// C# 12.0 features.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -160,6 +160,8 @@ internal override ObsoleteAttributeData ObsoleteAttributeData
get { return null; }
}

internal override int? TryGetOverloadResolutionPriority() => null;

bool ISynthesizedMethodBodyImplementationSymbol.HasMethodBodyDependency
{
get { return _getter.HasMethodBodyDependency; }
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -201,6 +201,8 @@ public FieldSymbol BackingField
get { return _backingField; }
}

internal override int? TryGetOverloadResolutionPriority() => null;

public override bool Equals(Symbol obj, TypeCompareKind compareKind)
{
if (obj == null)
Expand Down
5 changes: 5 additions & 0 deletions src/Compilers/CSharp/Portable/Symbols/ErrorMethodSymbol.cs
Original file line number Diff line number Diff line change
Expand Up @@ -294,5 +294,10 @@ internal sealed override bool HasAsyncMethodBuilderAttribute(out TypeSymbol buil
builderArgument = null;
return false;
}

internal sealed override int? TryGetOverloadResolutionPriority()
{
return null;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -92,5 +92,7 @@ public ErrorPropertySymbol(Symbol containingSymbol, TypeSymbol type, string name
public override ImmutableArray<PropertySymbol> ExplicitInterfaceImplementations { get { return ImmutableArray<PropertySymbol>.Empty; } }

public override ImmutableArray<CustomModifier> RefCustomModifiers { get { return ImmutableArray<CustomModifier>.Empty; } }

internal override int? TryGetOverloadResolutionPriority() => null;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -865,5 +865,10 @@ internal sealed override bool HasAsyncMethodBuilderAttribute(out TypeSymbol? bui
builderArgument = null;
return false;
}

internal sealed override int? TryGetOverloadResolutionPriority()
{
return null;
}
}
}
Loading