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

Interceptors: handle duplicates, additional signature validation, etc. #67786

Merged
merged 34 commits into from
Apr 25, 2023
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
5c4e289
Detect duplicate interceptions
RikkiGibson Apr 13, 2023
b769c0e
fix formatting
RikkiGibson Apr 13, 2023
a1f0eab
Fix dictionary usage. Test some file local attribute cases. Check sco…
RikkiGibson Apr 15, 2023
042d1d6
Fix nullable warning
RikkiGibson Apr 17, 2023
565c39a
Test 'params'. Test 'object.ReferenceEquals'. Handle more prototype c…
RikkiGibson Apr 17, 2023
f09ecb9
fix position diagnostics
RikkiGibson Apr 17, 2023
5edf9cf
fix lsp test
RikkiGibson Apr 18, 2023
b5fb444
Don't propagate InvokedAsExtensionMethod past lowering
RikkiGibson Apr 18, 2023
732dd91
Test file path case difference
RikkiGibson Apr 18, 2023
e47ce9d
Permit safe nullable differences
RikkiGibson Apr 19, 2023
678c4aa
Fix DiagnosticTest
RikkiGibson Apr 19, 2023
330ae61
Update spec
RikkiGibson Apr 19, 2023
aec866a
fix example type
RikkiGibson Apr 19, 2023
4b7fa44
Use explicit 'bool' type
RikkiGibson Apr 19, 2023
3534505
remove TODO
RikkiGibson Apr 19, 2023
d2053de
Fix crash when ROSLYN_TEST_USEDASSEMBLIES is defined
RikkiGibson Apr 19, 2023
d47412a
Fix nullable warning
RikkiGibson Apr 19, 2023
ca4cea8
Fix formatting
RikkiGibson Apr 19, 2023
21c1d56
More tests
RikkiGibson Apr 19, 2023
805fc9a
Add more tests
RikkiGibson Apr 20, 2023
87bcad6
Update baselines. Prefer reporting mismatch issues on attribute inste…
RikkiGibson Apr 20, 2023
3da2bad
Handle concurrent attribute decoding
RikkiGibson Apr 20, 2023
f4f05aa
add PROTOTYPE
RikkiGibson Apr 20, 2023
202914a
permit safe scoped variance
RikkiGibson Apr 20, 2023
c677ca7
Execute more test. Test intercepting ReferenceEquals.
RikkiGibson Apr 20, 2023
3a95687
Adjust comment
RikkiGibson Apr 20, 2023
2bfe8e2
adjust comment
RikkiGibson Apr 20, 2023
03be70a
skip verification in InterceptorExtern
RikkiGibson Apr 20, 2023
5c97905
add another duplicate test
RikkiGibson Apr 21, 2023
b6eb568
Clarify comment
RikkiGibson Apr 21, 2023
76d66d9
Add use site info
RikkiGibson Apr 21, 2023
f3c877e
fix double negative
RikkiGibson Apr 21, 2023
43205e8
test interceptable call with method type arguments in signature
RikkiGibson Apr 21, 2023
8664dad
add PROTOTYPE based on offline discussion
RikkiGibson Apr 25, 2023
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
14 changes: 7 additions & 7 deletions docs/features/interceptors.md
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ namespace System.Runtime.CompilerServices
`[InterceptsLocation]` attributes included in source are emitted to the resulting assembly, just like other custom attributes.

PROTOTYPE(ic): We may want to recognize `file class InterceptsLocationAttribute` as a valid declaration of the attribute, to allow generators to bring the attribute in without conflicting with other generators which may also be bringing the attribute in. See open question in [User opt-in](#user-opt-in).
https://github.com/dotnet/roslyn/issues/67079 is a bug which causes file-local source declarations of well-known attributes to be generally treated as known. When that bug is fixed, we may want to single out one or both of `InterceptableAttribute` and `InterceptsLocationAttribute` as "recognized, even though they are file-local".

#### File paths

Expand Down Expand Up @@ -111,17 +112,16 @@ Interceptors cannot have type parameters or be declared in generic types at any

### Signature matching

PROTOTYPE(ic): It is suggested to permit nullability differences and other comparable differences. Perhaps we can revisit the matching requirements of "partial methods" and imitate them here.

When a call is intercepted, the interceptor and interceptable methods must meet the signature matching requirements detailed below:
- When an interceptable instance method is compared to a classic extension method, we use the extension method in reduced form for comparison. The extension method parameter with the `this` modifier is compared to the instance method `this` parameter.
- The returns and parameters, including the `this` parameter, must have the same ref kinds and types, except that reference types with oblivious nullability can match either annotated or unannotated reference types.
- The returns and parameters, including the `this` parameter, must have the same ref kinds and types.
- A warning is reported instead of an error if a type difference is found where the types are not distinct to the runtime. For example, `string` and `string?`.
- Method names and parameter names are not required to match.
- Parameter default values are not required to match. When intercepting, default values on the interceptor method are ignored.
- `params` modifiers are not required to match.
- `scoped` modifiers and `[UnscopedRef]` must be equivalent.
- In general, attributes which normally affect the behavior of the call site, such as `[CallerLineNumber]` are ignored on the interceptor of an intercepted call.
- The only exception to this is when the attribute affects "capabilities" of the method in a way that affects safety, such as with `[UnscopedRef]`. In this case, attributes are required to match across interceptable and interceptor methods.
- The only exception to this is when the attribute affects "capabilities" of the method in a way that affects safety, such as with `[UnscopedRef]`. Such attributes are required to match across interceptable and interceptor methods.

Arity does not need to match between intercepted and interceptor methods. In other words, it is permitted to intercept a generic method with a non-generic interceptor.

Expand All @@ -133,7 +133,7 @@ If an `[InterceptsLocation]` attribute is found in the compilation which does no

### Interceptor accessibility

An interceptor must be accessible at the location where interception is occurring. PROTOTYPE(ic): This enforcement is not yet implemented.
An interceptor must be accessible at the location where interception is occurring.

An interceptor contained in a file-local type is permitted to intercept a call in another file, even though the interceptor is not normally *visible* at the call site.

Expand All @@ -157,7 +157,6 @@ During the binding phase, `InterceptsLocationAttribute` usages are decoded and t
- intercepted file-path and location
- attribute location
- attributed method symbol
PROTOTYPE(ic): the exact collection used to collect the attribute usages, and the exact way it is used, are not finalized. The main concern is to ensure we can scale to large numbers of interceptors without issue, and that we can report diagnostics for duplicate interception of the same location in a deterministic way.

At this time, diagnostics are reported for the following conditions:
- problems specific to the attributed interceptor method itself, for example, that it is not an ordinary method.
Expand All @@ -172,4 +171,5 @@ During the lowering phase, when a given `BoundCall` is lowered:

At this time, diagnostics are reported for the following conditions:
- incompatibility between the interceptor and interceptable methods, for example, in their signatures.
- *duplicate* `[InterceptsLocation]`, that is, multiple interceptors which intercept the same call. PROTOTYPE(ic): not yet implemented.
- *duplicate* `[InterceptsLocation]`, that is, multiple interceptors which intercept the same call.
- interceptor is not accessible at the call site.
20 changes: 19 additions & 1 deletion src/Compilers/CSharp/Portable/CSharpResources.resx
Original file line number Diff line number Diff line change
Expand Up @@ -7523,15 +7523,24 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ
<data name="ERR_InterceptorCharacterOutOfRange" xml:space="preserve">
<value>The given line is '{0}' characters long, which is fewer than the provided character number '{1}'.</value>
</data>
<data name="ERR_InterceptorLineCharacterMustBePositive" xml:space="preserve">
<value>Line and character numbers provided to InterceptsLocationAttribute must be positive.</value>
</data>
<data name="ERR_InterceptorPositionBadToken" xml:space="preserve">
<value>The provided line and character number does not refer to an interceptable method name, but rather to token '{0}'.</value>
</data>
<data name="ERR_InterceptorMustReferToStartOfTokenPosition" xml:space="preserve">
<value>The provided character number does not refer to the start of method name token '{0}'. Consider using character number '{1}' instead.</value>
<value>The provided line and character number does not refer to the start of token '{0}'. Did you mean to use line '{1}' and character '{2}'?</value>
</data>
<data name="ERR_InterceptorSignatureMismatch" xml:space="preserve">
<value>Cannot intercept method '{0}' with interceptor '{1}' because the signatures do not match.</value>
</data>
<data name="WRN_InterceptorSignatureMismatch" xml:space="preserve">
<value>Intercepting a call to '{0}' with interceptor '{1}', but the signatures do not match.</value>
</data>
<data name="WRN_InterceptorSignatureMismatch_Title" xml:space="preserve">
<value>Signatures of interceptable and interceptor methods do not match.</value>
</data>
<data name="ERR_InterceptableMethodMustBeOrdinary" xml:space="preserve">
<value>An interceptable method must be an ordinary member method.</value>
</data>
Expand All @@ -7553,6 +7562,15 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ
<data name="ERR_InterceptorNonUniquePath" xml:space="preserve">
<value>Cannot intercept a call in file with path '{0}' because multiple files in the compilation have this path.</value>
</data>
<data name="ERR_DuplicateInterceptor" xml:space="preserve">
<value>The indicated call is intercepted multiple times.</value>
</data>
<data name="ERR_InterceptorNotAccessible" xml:space="preserve">
<value>Cannot intercept call with '{0}' because it is not accessible within '{1}'.</value>
</data>
<data name="ERR_InterceptorScopedMismatch" xml:space="preserve">
<value>Cannot intercept call to '{0}' with '{1}' because of a difference in 'scoped' modifiers or '[UnscopedRef]' attributes.</value>
</data>
<data name="ERR_ConstantValueOfTypeExpected" xml:space="preserve">
<value>A constant value of type '{0}' is expected</value>
</data>
Expand Down
67 changes: 45 additions & 22 deletions src/Compilers/CSharp/Portable/Compilation/CSharpCompilation.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2247,44 +2247,39 @@ internal void AddModuleInitializerMethod(MethodSymbol method)
LazyInitializer.EnsureInitialized(ref _moduleInitializerMethods).Add(method);
}

private ConcurrentSet<(InterceptsLocationAttributeData, MethodSymbol)>? _interceptions;
// NB: the 'Many' case for these dictionary values means there are duplicates. An error is reported for this after binding.
private ConcurrentDictionary<(string FilePath, int Line, int Character), OneOrMany<(Location AttributeLocation, MethodSymbol Inteceptor)>>? _interceptions;
RikkiGibson marked this conversation as resolved.
Show resolved Hide resolved

internal void AddInterception(InterceptsLocationAttributeData location, MethodSymbol interceptor)
internal void AddInterception(string filePath, int line, int character, Location attributeLocation, MethodSymbol interceptor)
{
Debug.Assert(!_declarationDiagnosticsFrozen);
LazyInitializer.EnsureInitialized(ref _interceptions).Add((location, interceptor));

var dictionary = LazyInitializer.EnsureInitialized(ref _interceptions);
dictionary.AddOrUpdate((filePath, line, character),
addValueFactory: static (key, newValue) => OneOrMany.Create(newValue),
updateValueFactory: static (key, existingValue, newValue) => existingValue.Add(newValue),
factoryArgument: (attributeLocation, interceptor));
}

internal (InterceptsLocationAttributeData data, MethodSymbol interceptor)? GetInterceptor(Location? callLocation)
internal (Location AttributeLocation, MethodSymbol Interceptor)? TryGetInterceptor(Location? callLocation)
{
if (_interceptions is null || callLocation is null)
{
return null;
}

var sourceTree = callLocation.SourceTree;
Debug.Assert(sourceTree is not null);
var callLineColumn = callLocation.GetLineSpan().Span.Start;
foreach (var (interceptsLocation, interceptor) in _interceptions)
Debug.Assert(callLocation.SourceTree is not null);
var key = (callLocation.SourceTree.FilePath, callLineColumn.Line, callLineColumn.Character);
if (_interceptions.TryGetValue(key, out var oneInterception))
{
if (interceptsLocation.FilePath == sourceTree.FilePath
&& interceptsLocation.Line == callLineColumn.Line
&& interceptsLocation.Character == callLineColumn.Character)
{
return (interceptsLocation, interceptor);
}
// We don't expect to reach this phase if there are duplicate interceptors in the compilation.
return oneInterception.Single();
}

return null;
}

private void BuildInterceptionsMap()
{
// PROTOTYPE(ic): build a map where we can quickly lookup with a location and get a symbol.
// At this time, should report any duplicate interception diagnostics.
// NB: the attribute which appears lexically first wins a tie. Subsequent attributes referring to same location result in errors.
}

#endregion

#region Binding
Expand Down Expand Up @@ -3244,6 +3239,8 @@ internal override bool CompileMethods(
bool hasDeclarationErrors = !FilterAndAppendDiagnostics(diagnostics, GetDiagnostics(CompilationStage.Declare, true, cancellationToken), excludeDiagnostics, cancellationToken);
excludeDiagnostics?.Free();

hasDeclarationErrors |= CheckDuplicateInterceptions(diagnostics);
Copy link
Member

@jcouv jcouv Apr 24, 2023

Choose a reason for hiding this comment

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

From offline discussion, maybe this should be moved into CompileMethodBodies and maybe CheckDuplicateFilePaths as well.
Then the ERR_ModuleEmitFailure reported in TryGetInterceptor becomes unnecessary.
That's okay as a PROTOTYPE

Copy link
Member

Choose a reason for hiding this comment

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

Also consider adding tests for duplicate interceptions and duplicate file paths in a compilation for ref-only assembly (showing whether or not we ran those checks).

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This comment also seems relevant.

#67786 (comment)

Basically, the search is for a consistent place to report duplicates at the end of the declaration phase. This place would be appropriate for both duplicate file-local types and for interceptors.


// TODO (tomat): NoPIA:
// EmbeddedSymbolManager.MarkAllDeferredSymbolsAsReferenced(this)

Expand Down Expand Up @@ -3275,8 +3272,6 @@ internal override bool CompileMethods(
return false;
}

BuildInterceptionsMap();

// Perform initial bind of method bodies in spite of earlier errors. This is the same
// behavior as when calling GetDiagnostics()

Expand Down Expand Up @@ -3379,12 +3374,40 @@ public override void VisitNamedType(NamedTypeSymbol symbol)
}
}

/// <returns><see langword="true"/> if file types are present in files with duplicate file paths. Otherwise, <see langword="false" />.</returns>
private bool CheckDuplicateFilePaths(DiagnosticBag diagnostics)
{
var visitor = new DuplicateFilePathsVisitor(diagnostics);
return visitor.CheckDuplicateFilePathsAndFree(SyntaxTrees, GlobalNamespace);
}

/// <returns><see langword="true"/> if duplicate interceptors are present in the compilation. Otherwise, <see langword="false" />.</returns>
private bool CheckDuplicateInterceptions(DiagnosticBag diagnostics)
Copy link
Contributor Author

Choose a reason for hiding this comment

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

I am wondering if it would make sense to call this in SourceModuleSymbol.ForceComplete, for example, after we find this.GlobalNamespace.HasComplete(CompletionPart.MembersCompleted). At this point I think we should have reported declaration diagnostics for all members, and we would know if there are duplicate interceptors for a location.

{
if (_interceptions is null)
{
return false;
}

var anyDuplicates = false;
RikkiGibson marked this conversation as resolved.
Show resolved Hide resolved
foreach ((_, OneOrMany<(Location, MethodSymbol)> interceptionsOfAGivenLocation) in _interceptions)
{
Debug.Assert(interceptionsOfAGivenLocation.Count != 0);
if (interceptionsOfAGivenLocation.Count == 1)
{
continue;
}

anyDuplicates = true;
foreach (var (attributeLocation, _) in interceptionsOfAGivenLocation)
{
diagnostics.Add(ErrorCode.ERR_DuplicateInterceptor, attributeLocation);
}
}

return anyDuplicates;
}

private void GenerateModuleInitializer(PEModuleBuilder moduleBeingBuilt, DiagnosticBag methodBodyDiagnosticBag)
{
Debug.Assert(_declarationDiagnosticsFrozen);
Expand Down
6 changes: 6 additions & 0 deletions src/Compilers/CSharp/Portable/Errors/ErrorCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2209,6 +2209,12 @@ internal enum ErrorCode
ERR_InterceptorFilePathCannotBeNull = 27013,
ERR_InterceptorNameNotInvoked = 27014,
ERR_InterceptorNonUniquePath = 27015,
ERR_DuplicateInterceptor = 27016,
WRN_InterceptorSignatureMismatch = 27017,
ERR_InterceptorNotAccessible = 27018,
ERR_InterceptorScopedMismatch = 27019,
ERR_InterceptorLineCharacterMustBePositive = 27020,

#endregion

// Note: you will need to do the following after adding warnings:
Expand Down
6 changes: 6 additions & 0 deletions src/Compilers/CSharp/Portable/Errors/ErrorFacts.cs
Original file line number Diff line number Diff line change
Expand Up @@ -531,6 +531,7 @@ internal static int GetWarningLevel(ErrorCode code)
case ErrorCode.WRN_CapturedPrimaryConstructorParameterPassedToBase:
case ErrorCode.WRN_UnreadPrimaryConstructorParameter:
case ErrorCode.WRN_CallNotInterceptable:
case ErrorCode.WRN_InterceptorSignatureMismatch:
return 1;
default:
return 0;
Expand Down Expand Up @@ -584,6 +585,10 @@ internal static bool IsBuildOnlyDiagnostic(ErrorCode code)
case ErrorCode.ERR_InterceptorSignatureMismatch:
case ErrorCode.ERR_InterceptorMustHaveMatchingThisParameter:
case ErrorCode.ERR_InterceptorMustNotHaveThisParameter:
case ErrorCode.ERR_DuplicateInterceptor:
case ErrorCode.WRN_InterceptorSignatureMismatch:
case ErrorCode.ERR_InterceptorNotAccessible:
case ErrorCode.ERR_InterceptorScopedMismatch:
// Update src\EditorFeatures\CSharp\LanguageServer\CSharpLspBuildOnlyDiagnostics.cs
// whenever new values are added here.
return true;
Expand Down Expand Up @@ -2326,6 +2331,7 @@ internal static bool IsBuildOnlyDiagnostic(ErrorCode code)
case ErrorCode.ERR_InterceptorFilePathCannotBeNull:
case ErrorCode.ERR_InterceptorNameNotInvoked:
case ErrorCode.ERR_InterceptorNonUniquePath:
case ErrorCode.ERR_InterceptorLineCharacterMustBePositive:
case ErrorCode.ERR_ConstantValueOfTypeExpected:
case ErrorCode.ERR_UnsupportedPrimaryConstructorParameterCapturingRefAny:
return false;
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading