-
Notifications
You must be signed in to change notification settings - Fork 1.6k
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
Shared, type inference: Add inference for type parameters with constraints (base type mentions) #19102
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Copilot wasn't able to review any files in this pull request.
Files not reviewed (1)
- shared/typeinference/codeql/typeinference/internal/TypeInference.qll: Language not supported
Tip: Copilot only keeps its highest confidence comments to reduce noise and keep you focused. Learn more
DCA is not looking good. I think maybe the change to |
c6ad485
to
ba9edf8
Compare
I can't get DCA to work right now, but when running locally on Edit: DCA is working again and the results are now fine. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Impressive work, very nice! My comments are mostly minor; one thing that I note is that by introducing the pathToSub
generalization, we add even more assumptions about covariance, but we already have a follow-up task about handling contravariance properly.
* of `Mid`, | ||
* - ``C`1`` is mentioned at `0` for immediate base type mention `Mid<C<T4>>` | ||
* - ``C`1`` is mentioned at `T3` for immediate base type mention `Mid<C<T4>>` | ||
* of `Sub`, and |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove and
* of `Sub`, and | ||
* - ``C`1`` is mentioned at `0` and implicitly at `0.0` for transitive base type | ||
* - `T4` is mentioned at `T3.T1` for immediate base type mention `Mid<C<T4>>` | ||
* of `Sub`, and |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove and
* - ``C`1`` is mentioned at `0` and implicitly at `0.0` for transitive base type | ||
* - `T4` is mentioned at `T3.T1` for immediate base type mention `Mid<C<T4>>` | ||
* of `Sub`, and | ||
* - ``C`1`` is mentioned at `T2` and implicitly at `T2.T1` for transitive base type | ||
* mention `Base<C<T3>>` of `Sub`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Replace .
with , and
exists(Declaration target | | ||
/** | ||
* Holds if inferring types at `a` might depend on the type at `apos` | ||
* having `baseMention` as a transitive base type mention. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should be
having `base` as a transitive base type.
not t = immediateBase.getATypeParameter() and | ||
baseTypeMentionHasTypeAt(immediateBase, baseMention, path, t) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This pattern (and the dual) appears twice, so I think it perhaps still makes sense to have the predicates baseTypeMentionHasNonTypeParameterAt
and baseTypeMentionHasTypeParameterAt
, which then simply call baseTypeMentionHasTypeAt
and add appropriate restrictions.
* For this example | ||
* ```csharp | ||
* interface IFoo<A> { } | ||
* void M<T1, T2>(T2 item) where T2 : IFoo<T1> { } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps change the return type from void
to T1
, to make it an example where type inference would actually be relevant.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good idea!
* - `path1 = ""`, | ||
* - `tp1 = T2`, | ||
* - `constraint = IFoo`, | ||
* - `path2 = "A"`, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
,
-> , and
* - `tp1 = T2`, | ||
* - `constraint = IFoo`, | ||
* - `path2 = "A"`, | ||
* - `tp2 = T1` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add a final .
* `base`. For example, in | ||
* Holds if `baseMention` is a (transitive) base type mention of the | ||
* type of `a` at position `apos` at path `pathToSub`, and `t` is | ||
* mentioned (implicitly) at `path` inside `base`. For example, in | ||
* | ||
* ```csharp |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The example should mention that pathToSub = ""
. Perhaps it would also be worth adding an example where pathToSub != ""
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For now I've just mentioned pathToSub = ""
. I think adding another example is a good idea. Now that relevantAccess
is added perhaps the existing example needs revamping as well? Now the predicate only finds the supertypes when necessary, and ToString
probably doesn't need to do that?
t = immediateBaseMention.resolveTypeAt(path0) and | ||
path0.isCons(tp, suffix) and | ||
path = prefix.append(suffix) | ||
) | ||
) | ||
) | ||
} | ||
|
||
/** Similar to `baseTypeMentionHasTypeAt` but FIXME: */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should be fixed ;-)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fine to simply the remove the QL doc.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oops 😅
@@ -514,7 +527,7 @@ module Make1<LocationSig Location, InputSig1<Location> Input1> { | |||
private module AccessBaseType { | |||
/** | |||
* Holds if inferring types at `a` might depend on the type at `path` of | |||
* `apos` having `baseMention` as a transitive base type mention. | |||
* `apos` having `base` as a transitive base type mention. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
remove mention
.
The first commit in this PR is a small refactor that tries to:
baseTypeMentionHasTypeParameterAt
andbaseTypeMentionHasNonTypeParameterAt
into a single predicate, to cut down some duplication.hasBaseTypeMention
by restrict theTypeMention
s that it considers. On the type inference test database this refactor changes the number of tuples from126 + 381 + 66 = 573
to0 + 62 + 0 = 62
in that predicate, with no change to the results.In addition the PR improves type inference for calls that target functions with type parameters that have trait bounds. For instance, when calling a function like
the new
typeParameterConstraintHasTypeParameter
handles inferringT1
. It does this by noticingT1
appears inside another type parameterT2
,T2
occurs inside the declared type ofx
at the pathT
,T
as a base type mention ofMyTrait<T1>
and inferringT1
from that.