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

Shared, type inference: Add inference for type parameters with constraints (base type mentions) #19102

Merged
merged 8 commits into from
Mar 28, 2025

Conversation

paldepind
Copy link
Contributor

@paldepind paldepind commented Mar 24, 2025

The first commit in this PR is a small refactor that tries to:

  • Combine baseTypeMentionHasTypeParameterAt and baseTypeMentionHasNonTypeParameterAt into a single predicate, to cut down some duplication.
  • Reduce the number of tuples in hasBaseTypeMention by restrict the TypeMentions that it considers. On the type inference test database this refactor changes the number of tuples from 126 + 381 + 66 = 573 to 0 + 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

struct MyThing<T> { ... }
fn call_trait_thing_m1<T1, T2: MyTrait<T1>>(x: MyThing<T2>) -> T1 { ... }

the new typeParameterConstraintHasTypeParameter handles inferring T1. It does this by noticing

  • that T1 appears inside another type parameter T2,
  • that T2 occurs inside the declared type of x at the path T,
  • considering the type of the argument at T as a base type mention of MyTrait<T1> and inferring T1 from that.

@paldepind paldepind added the Rust Pull requests that update Rust code label Mar 24, 2025
@paldepind paldepind marked this pull request as ready for review March 24, 2025 14:08
@Copilot Copilot bot review requested due to automatic review settings March 24, 2025 14:08
Copy link
Contributor

@Copilot Copilot AI left a 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

@paldepind paldepind added the no-change-note-required This PR does not need a change note label Mar 24, 2025
@paldepind paldepind requested a review from hvitved March 24, 2025 14:44
@paldepind
Copy link
Contributor Author

DCA is not looking good. I think maybe the change to declarationBaseType made that predicate grow in size. I'll investigate.

@paldepind paldepind removed the request for review from hvitved March 24, 2025 17:16
@github-actions github-actions bot removed the Rust Pull requests that update Rust code label Mar 24, 2025
@paldepind paldepind added the Rust Pull requests that update Rust code label Mar 25, 2025
@paldepind paldepind changed the title Shared: Refactor type inference Shared, type inference: Add inference for type parameters with constraints (base type mentions) Mar 25, 2025
@paldepind paldepind requested a review from hvitved March 25, 2025 15:16
@paldepind
Copy link
Contributor Author

paldepind commented Mar 25, 2025

I can't get DCA to work right now, but when running locally on rust it seems that the bad join is fixed, so I think this is ready for review 😃

Edit: DCA is working again and the results are now fine.

Copy link
Contributor

@hvitved hvitved left a 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
Copy link
Contributor

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
Copy link
Contributor

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`.
Copy link
Contributor

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.
Copy link
Contributor

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.

Comment on lines 309 to 310
not t = immediateBase.getATypeParameter() and
baseTypeMentionHasTypeAt(immediateBase, baseMention, path, t)
Copy link
Contributor

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> { }
Copy link
Contributor

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.

Copy link
Contributor Author

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"`,
Copy link
Contributor

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`
Copy link
Contributor

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
Copy link
Contributor

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 != ""?

Copy link
Contributor Author

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: */
Copy link
Contributor

Choose a reason for hiding this comment

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

Should be fixed ;-)

Copy link
Contributor

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.

Copy link
Contributor Author

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.
Copy link
Contributor

Choose a reason for hiding this comment

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

remove mention.

@paldepind paldepind merged commit 884c828 into github:main Mar 28, 2025
36 of 37 checks passed
@paldepind paldepind deleted the rust-ti-refactor branch March 28, 2025 13:02
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
no-change-note-required This PR does not need a change note Rust Pull requests that update Rust code
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants