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

best_blame_constraint: Blame better constraints when the region graph has cycles from invariance or 'static #133858

Merged
merged 12 commits into from
Jan 8, 2025

Conversation

dianne
Copy link
Contributor

@dianne dianne commented Dec 4, 2024

This fixes #132749 by changing which constraint is blamed for region errors in several cases. best_blame_constraint had a heuristic that tried to pinpoint the constraint causing an error by filtering out any constraints where the outliving region is unified with the ultimate target region being outlived. However, it used the SCCs of the region graph to do this, which is unreliable; in particular, if the target region is 'static, or if there are cycles from the presence of invariant types, it was skipping over the constraints it should be blaming. As is the case in that issue, this could lead to confusing diagnostics. The simplest fix seems to work decently, judging by test stderr: this makes best_blame_constraint no longer filter constraints by their outliving region's SCC.

There are admittedly some quirks in the test output. In many cases, subdiagnostics that depend on the particular constraint being blamed have either started or stopped being emitted. After starting at this for quite a while, I think anything too fickle about whether it outputs based on the particular constraint being blamed should instead be looking at the constraint path as a whole, similar to what's done for the placeholder-from-predicate note.

Very many tests involving invariant types gained a note pointing out the types' invariance, but in a few cases it was lost. A particularly illustrative example is tests/ui/lifetimes/copy_modulo_regions.stderr; I'd argue the new constraint is a better one to blame, but it lacks the variance diagnostic information that's elsewhere in the constraint path. If desired, I can try making that note check the whole path rather than just the blamed constraint.

The subdiagnostic BorrowExplanation::add_object_lifetime_default_note depends on a Cast being blamed, so a special case was necessary to keep it from disappearing from tests specifically testing for it. However, see the FIXME comment in that commit; I think the special case should be removed once that subdiagnostic works properly, but it's nontrivial enough to warrant a separate PR. Incidentally, this removes the note from a test where it was being added erroneously: in tests/ui/borrowck/two-phase-surprise-no-conflict.stderr, the object lifetime is explicitly provided and it's not 'static.

@rustbot
Copy link
Collaborator

rustbot commented Dec 4, 2024

r? @nnethercote

rustbot has assigned @nnethercote.
They will have a look at your PR within the next two weeks and either review your PR or reassign to another reviewer.

Use r? to explicitly pick a reviewer

@rustbot rustbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. labels Dec 4, 2024
@nnethercote
Copy link
Contributor

I am not a good person to review nitty gritty borrowck error message changes :) Let's try someone else who is probably more appropriate: r? @lcnr

this may have perf consequence

Let's check: @bors try @rust-timer queue

@rust-timer

This comment has been minimized.

@rustbot rustbot assigned lcnr and unassigned nnethercote Dec 4, 2024
@rustbot rustbot added the S-waiting-on-perf Status: Waiting on a perf run to be completed. label Dec 4, 2024
@bors
Copy link
Contributor

bors commented Dec 4, 2024

⌛ Trying commit 189b2f8 with merge 889e047...

bors added a commit to rust-lang-ci/rust that referenced this pull request Dec 4, 2024
…static, r=<try>

`best_blame_constraint`: Blame better constraints when failing to outlive `'static`

This fixes rust-lang#132749 by changing which constraint is blamed for region errors when failing to prove a region outlives `'static`. The comments give a better explanation, but very roughly, all the `'static: R` edges in the region graph made `best_blame_constraint` consider every constraint (other than those from ascriptions, returns, and yields) uninteresting to point to, so it chose semi-randomly based on an ordering of how interesting each variant of `ConstraintCategory` is expected to be. This PR (admittedly hackily) makes it ignore all those edges, so that the existing logic works the same as it would when failing to outlive any other region.

Looking at UI test diffs, most of them that changed I think changed for the better. Unfortunately, since a lot of borrowck's diagnostics depend on exactly which constraint is blamed, some things broke. A list of what I'm not happy with:
- For `CopyBound` constraints, e.g. [`tests/ui/nll/do-not-ignore-lifetime-bounds-in-copy.stderr`](https://github.com/rust-lang/rust/compare/master...dianne:rust:better-blame-constraints-for-static?expand=1#diff-e220f1e433c5e48d9afd431787f6ff27fc66b653762ee8a0283e370c2d88e7d0), I think it's helpful to surface that the `Copy` impl introduces the bound. If this is an issue, maybe it's worth prioritizing it or adding a variant of `ExtraConstraintInfo` for it.
- Likewise for `UseAsConst` and `UseAsStatic`; I've already added a special case for those. Without a special case, this was blaming `CallArgument` in [`tests/ui/consts/const-mut-refs/mut_ref_in_final.stderr`](https://github.com/dianne/rust/blob/189b2f892e6d0809a77fc92fe1108a07d8de9be0/tests/ui/consts/const-mut-refs/mut_ref_in_final.stderr#L38), which seemed pretty egregious. I'm assuming we want to blame `UseAsConst`/`UseAsStatic` when possible, rather than add something to `ExtraConstraintInfo`, since it's pretty unambiguously to-blame for "outlives `'static`" constraints. The special-casing admittedly also changes the output for [`tests/ui/inline-const/const-match-pat-lifetime-err.rs`](https://github.com/rust-lang/rust/compare/master...dianne:rust:better-blame-constraints-for-static?expand=1#diff-e4d2c147aa96dd8dd963ec3d98ead9a8096c9de809d19ab379be3c53951ca1ca), but I think the new output there is fine; it's more direct about how `'a` and `'static` are getting related.
- The subdiagnostic [`BorrowExplanation::add_object_lifetime_default_note`](https://doc.rust-lang.org/nightly/nightly-rustc/rustc_borrowck/diagnostics/explain_borrow/enum.BorrowExplanation.html#method.add_object_lifetime_default_note) depends on `Cast` being reported as the constraint category, so I've added a special case for it to keep it disappearing from [`tests/ui/traits/trait-object-lifetime-default-note.stderr`](https://github.com/dianne/rust/blob/better-blame-constraints-for-static/tests/ui/traits/trait-object-lifetime-default-note.stderr)[^1] . As I've outlined in a `FIXME` comment though, I think that subdiagnostic needs changing. There's multiple cases throughout `tests/ui` where it would be helpful, but doesn't fire, because a different constraint is picked. rust-lang#131008 would also benefit from that note, but there's not a coercion there at all. I tried making it fire in more cases, but fundamentally, since it doesn't *actually* check if an object lifetime default was used in the HIR, it produced some extraneous notes[^2]. I tried seeing if it'd be easy enough to fix that first, but it seems nontrivial[^3] enough to warrant a separate PR.

A final note: this may have perf consequences, since `best_blame_constraint` gets called on happy code too. If it's an issue, I can try to make it faster, or only do the expensive stuff when an error's been hit, or something. If nothing else, this makes the fallback logic (still relevant for invariant lifetimes[^4]) a bit faster.

[^1]: Incidentally, without the special-casing, this would pick `CallArgument`, which I think produces a nicer message, apart from the missing note.

[^2]: There's even some in current UI test output: [`tests/ui/borrowck/two-phase-surprise-no-conflict.stderr`](https://github.com/rust-lang/rust/blob/733616f7236b4be140ce851a30b3bb06532b9364/tests/ui/borrowck/two-phase-surprise-no-conflict.stderr#L68). The object lifetime is [explicitly provided](https://github.com/rust-lang/rust/blob/733616f7236b4be140ce851a30b3bb06532b9364/tests/ui/borrowck/two-phase-surprise-no-conflict.rs#L94), and despite the note, it's not `'static`.

[^3]: In case anyone reading this has advice: ultimately I think there has to be a choice between extraneous notes and missing notes unless "this object type had a missing lifetime in the HIR" is in crate metadata for some reason, since rustdoc doesn't elaborate object lifetime defaults, and the note is potentially helpful for library-users who may be wondering where a `'static` lifetime came from. That said, it's a bit confusing to point out lifetime defaults when they're not relevant[^5], so it's at least worth a best effort to look at user annotations. However, the HIR missing an object lifetime could be anywhere: a cast, an ascription, in the return type, in an input, in the signature for a function that's called, in an ADT field, in a type alias... I've considered looking at the entire output of `RegionInferenceContext::find_constraint_path_between_regions` and doing a best-effort association between typeck results (ideally MIR typeck for the region information) and whatever HIR seems relevant. But that seems like a lot of work for that note.

[^4]: Cycles in the region graph due to invariance also confuse `better_blame_constraint`'s attempt to pick a constraint where the regions aren't unified. I'm not sure if there's a better heuristic to use there, though; I played around a bit with it, but my experiments only made diagnostic output worse.

[^5]: Unless maybe there's a way of rewording the note so that it always makes sense to output when object lifetimes are involved in region errors?
@bors
Copy link
Contributor

bors commented Dec 4, 2024

☀️ Try build successful - checks-actions
Build commit: 889e047 (889e047c4bb3d82cd34c0c703f2bd13964c828ca)

@rust-timer

This comment has been minimized.

@rust-timer
Copy link
Collaborator

Finished benchmarking commit (889e047): comparison URL.

Overall result: no relevant changes - no action needed

Benchmarking this pull request likely means that it is perf-sensitive, so we're automatically marking it as not fit for rolling up. While you can manually mark this PR as fit for rollup, we strongly recommend not doing so since this PR may lead to changes in compiler perf.

@bors rollup=never
@rustbot label: -S-waiting-on-perf -perf-regression

Instruction count

This benchmark run did not return any relevant results for this metric.

Max RSS (memory usage)

Results (secondary 2.4%)

This is a less reliable metric that may be of interest but was not used to determine the overall result at the top of this comment.

mean range count
Regressions ❌
(primary)
- - 0
Regressions ❌
(secondary)
2.4% [2.4%, 2.4%] 1
Improvements ✅
(primary)
- - 0
Improvements ✅
(secondary)
- - 0
All ❌✅ (primary) - - 0

Cycles

This benchmark run did not return any relevant results for this metric.

Binary size

This benchmark run did not return any relevant results for this metric.

Bootstrap: 769.678s -> 767.43s (-0.29%)
Artifact size: 330.86 MiB -> 330.84 MiB (-0.01%)

@rustbot rustbot removed the S-waiting-on-perf Status: Waiting on a perf run to be completed. label Dec 5, 2024
Copy link
Contributor

@lcnr lcnr left a comment

Choose a reason for hiding this comment

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

I think this mostly makes sense to me, it has been somewhat difficult to review for me however. It would be really nice if you could clean up the surrounding code for a bite, even though it's not trictly part of your change. This would make reviewing this code a lot more pleasant.

  • afaict extra_info is only mutated by a single push in this function, so 1) why are we using a Vec instead of an Option here and 2) could we its computation into a separate function
  • I can't immediately tell what changes due to removing the 'whatever: 'static edges and what changes due to the explicit check+eager return, can you give me some more info here, or maybe even split it into 2 commits?

Comment on lines 2192 to 2197
// FIXME(dianne): `BorrowExplanation::add_object_lifetime_default_note` depends on a
// coercion being blamed, so revert to the old blaming logic to prioritize that.
// The note's logic should be reworked, though; it's flaky (#131008 doesn't have a
// coercion, and even with this hack, one isn't always blamed when present).
// Only checking for a coercion also makes the note appear where it shouldn't
// shouldn't (e.g. `tests/ui/borrowck/two-phase-surprise-no-conflict.stderr`).
Copy link
Contributor

Choose a reason for hiding this comment

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

from what I can tell, this FIXME is stating that this branch is causing us to not emit a add_object_lifetime_default_note in cases where we should. Please link to an affected test

Copy link
Contributor Author

Choose a reason for hiding this comment

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

ah, the wording of the FIXME is unclear; I'll see if I can make it better. I've made sure to still emit that note where it was previously emitted. that branch is a hack specifically to keep the note present in tests/ui/traits/trait-object-lifetime-default-note.stderr, since otherwise the CallArgument constraint would be blamed and the note would disappear; it'll be more apparent once I get rid of the UseAsConst/UseAsStatic special-casing and split the Cast case into a separate commit. I'll refer to that test in the FIXME too.

the missing notes the FIXME is referring to are cases where the note would make sense (and imo should be emitted)1, but it wasn't emitted even before this change. the note depends on a coercion being blamed, but it won't always be blamed for region errors from object lifetime defaults (nor will a coercion necessarily be present). handling that properly is (afaict) nontrivial; I'd be happy to do what I can there in a separate PR, but I might either need some help with wording or some direction on what heuristics would make sense, since doing it as perfectly as possible seems quite involved.

Footnotes

  1. see GAT type Assoc<T: ?Sized> implicitly requires Self to be 'static #131008, which would probably need a fairly general solution. I remember finding in-tree tests that are missing the note too though (which would require a less-general but still nontrivial fix), so I can dig those up if it'd help

// If an "outlives 'static" constraint was from use as a const or static, blame that.
if target_is_static
&& blame_source
&& let Some(old_best) = categorized_path.iter().min_by_key(|p| p.category)
Copy link
Contributor

Choose a reason for hiding this comment

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

this feels odd to me, do you want to disable this if there's also one step in the path whose category is a ConstraintCategory::Return or would it be pattern to walk the path to detect whether any ConstraintCategory::UseAsConst exists?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

looking at it again, that special case shouldn't be there at all; it's just papering over a more fundamental issue. from what I can tell, the reason tests/ui/consts/const-mut-refs/mut_ref_in_final.stderr needs the special-casing is that UnsafeCell<T> is invariant in T, hence the outliving region of the UseAsConst/UseAsStatic constraint is in the same SCC as the target, hence it gets skipped. for a quick workaround, this could check if there's a constraint on the path with an invariant variance_info before removing any edges from the region graph. it's sort of justifiable, since back-edges from invariance would also need to be removed in order to get the properties I wanted from pruning the 'static: R edges. that said, I think this should handle invariance properly if it can. leaving it sometimes-wrong feels pretty unsatisfying, especially if a proper fix might handle this whole issue cleanly. I'll have to look into it further.

@dianne
Copy link
Contributor Author

dianne commented Dec 11, 2024

  • afaict extra_info is only mutated by a single push in this function, so 1) why are we using a Vec instead of an Option here and 2) could we its computation into a separate function

I think the intent is for there to possibly be more variants of ExtraConstraintInfo. I agree though; think it'd be cleaner to move it to a separate function and get rid of ExtraConstraintInfo entirely. I'll try that!

  • I can't immediately tell what changes due to removing the 'whatever: 'static edges and what changes due to the explicit check+eager return, can you give me some more info here, or maybe even split it into 2 commits?

no problem! I'll split it up; that'll help me too. I'll also be removing the special-case for UseAsConst/UseAsStatic as per my reply to your review comment, so hopefully that'll remove some noise.

@dianne dianne force-pushed the better-blame-constraints-for-static branch from 189b2f8 to 364ca7f Compare December 12, 2024 17:14
@dianne
Copy link
Contributor Author

dianne commented Dec 12, 2024

I've pushed some changes:

  • The first two commits are cleanup of the surrounding code.
  • The third is the primary diagnostic change, which I've reworked. I'll try and write up a revised description in the main PR text (edit: it's there now), but in short, I've removed all of the logic to filter constraints by their sup's SCC. The result is simpler and I think it produces better diagnostics (though a few bits could use some work; I give my opinion in the longer writeup).
  • The fourth commit is the special case for tests/ui/traits/trait-object-lifetime-default-note.rs, including an updated FIXME comment.

@dianne dianne changed the title best_blame_constraint: Blame better constraints when failing to outlive 'static best_blame_constraint: Blame better constraints when the region graph has cycles from invariance or 'static Dec 12, 2024
@bors
Copy link
Contributor

bors commented Dec 13, 2024

☔ The latest upstream changes (presumably #132706) made this pull request unmergeable. Please resolve the merge conflicts.

Copy link
Contributor

@lcnr lcnr left a comment

Choose a reason for hiding this comment

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

r=me on the first two commits, some of the changes look like they make things worse 🤔 have to look into it in more detail

thanks for splitting this PR into these commits, I feel like it's a lot more pleasant to review ❤️

| ^^^^ - - let's call the lifetime of this reference `'2`
| | |
| | let's call the lifetime of this reference `'1`
| assignment requires that `'1` must outlive `'2`
Copy link
Contributor

Choose a reason for hiding this comment

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

this seems wrong to me 🤔

the assignment is very clearly not part of the function signature xx that's likely caused by the async fn desugaring first moving bufs into a local, and we then constrain the type of the local to be &'a mut &'a :/

I think we should somehow filter this, I feel like the current approach of figuring out what to blame is generally a bit 🤷 i would expect us to instead iterate over all constraints and take the "best one" (while preferring earlier ones), instead of simply taking the first interesting one

Copy link
Contributor Author

Choose a reason for hiding this comment

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

ah, yeah, that's definitely wrong. I had a note about that test but I completely forgot to address it; thank you for the catch! I've added a commit that makes it try not to blame constraints from desugarings. it includes a special case to allow blaming returns from ?, since otherwise tests/ui/async-await/issue-67765-async-diagnostic.stderr would be less clear about why the usage Err(b) is problematic (and ? including a return is a lot more clear than the async sugar including an assignment). I'm not a desugaring expert, though, so I'm not sure if any others are intuitive/meaningful enough to be blamed (possibly with special wording?).

with regard to iterating over all constraints and picking the "best one", I agree that it should look at all the constraints in some way, but I'll have to think about if there's a better heuristic for the single best than position on the path (with some filter). I think returns are generally good to point to, and CopyBound constraints should maybe at least get an honorable mention in a note (if not the full blame) since they're subtle, but beyond that I think it gets a bit fuzzy. e.g. ascriptions and casts are helpful to point to when they syntactically refer to a relevant type/region, but are much less helpful if they leave it up to inference. for another example, UseAsConst is sometimes the best we've got at the top level, but within function bodies it's usually less helpful. besides the category, it feels like it should be possible to use constraints' regions to pinpoint the most blameable constraint, but I haven't found a straightforward way to get blame out of the constraint graph or region graph, given the cycles. constraints' location/span are helpful, but mostly once we're in proper diagnostic code and can look at types and the HIR and MIR and such. the compromise I've imagined is to pick a good-enough-but-maybe-not-best constraint to blame, and anything important enough to surface in addition to that can be picked out from the constraint path by diagnostics (though of course if there's some other way to get the blamed constraint as good as possible too, that'd be ideal).

@dianne dianne force-pushed the better-blame-constraints-for-static branch from 259b930 to 8e4fcc4 Compare December 13, 2024 19:36
@bors
Copy link
Contributor

bors commented Dec 14, 2024

☔ The latest upstream changes (presumably #134292) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the better-blame-constraints-for-static branch from 8e4fcc4 to 326a2bc Compare December 14, 2024 08:24
Copy link
Contributor

@lcnr lcnr left a comment

Choose a reason for hiding this comment

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

a final suggestion, apart from this r=me 🤔

borrowck diagnostics are :/

@@ -1,13 +1,15 @@
error: lifetime may not live long enough
--> $DIR/ex3-both-anon-regions-2.rs:2:5
--> $DIR/ex3-both-anon-regions-2.rs:1:14
|
LL | fn foo(&mut (ref mut v, w): &mut (&u8, &u8), x: &u8) {
Copy link
Contributor

Choose a reason for hiding this comment

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

why do we point at ref mut v when talking about "assignment" here?

|
LL | impl<'b> S<'b> {
| -- lifetime `'b` defined here
LL | fn bar<'a>(&'a mut self) -> &'a mut &'a i32 {
| -- lifetime `'a` defined here
LL | match self.0 { ref mut x => x }
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ method was supposed to return data with lifetime `'b` but it is returning data with lifetime `'a`
| ^^^^^^^^^ assignment requires that `'a` must outlive `'b`
Copy link
Contributor

Choose a reason for hiding this comment

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

okay, we do we consider ref mut x patterns to be an assignment?

Copy link
Contributor

Choose a reason for hiding this comment

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

I guess this is a preexisting issue? Though maybe we should never consider this to be an interesting node?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

oh, I missed that! I believe those bindings are "assignment"s because once they're lowered to MIR locals, they're assigned to by MIR assignment statements; the constraint generated by that is what's getting blamed. afaict currently there's not enough information in the MIR to efficiently make that distinction when assigning constraint categories. but we do have a span, so it shouldn't be too hard to call assignments "binding"s in diagnostics when syntactically they feel more like a "binding" than an "assignment".

that said, those are the wrong constraints to blame for these tests; these are both regressions, though it's just by chance they were correct in the first place. I assume because the error depends on &mut being invariant in its parameter, we're looking at things kind of backwards1 and we picked a constraint from the unhelpful end of the path. I think it should be possible to detect when constraints arising from contravariance are being followed and flip the path around if so.. it'll probably make for better diagnostics, but I'll admit I don't love relying so heavily on what direction we traverse the constraint graph in for this. I'm worried something more robust might require collecting more information in borrowck, but I'll look into it at least.

sorry for the huge number of very specific stderr changes, and thank you for taking the time to review this. I definitely agree that it's :/

Footnotes

  1. this is present in this test when blaming the correct constraint too: the label "method was supposed to return data with lifetime 'b but it is returning data with lifetime 'a" is peculiar when the return type in the signature has lifetime 'a and the body has lifetime 'b. I think that message was worded assuming return types are always covariant. it should be fixed too (maybe with a few other tweaks as well.. ideally it'd suggest replacing 'a with 'b rather than adding 'a: 'b, like we do for 'static), but probably that should go in a different PR.

Copy link
Contributor

Choose a reason for hiding this comment

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

hmm, one approach would be to change this code here to consider more assignments to be boring:

let category = match place.as_local() {
Some(RETURN_PLACE) => {
let defining_ty = &self.universal_regions.defining_ty;
if defining_ty.is_const() {
if tcx.is_static(defining_ty.def_id()) {
ConstraintCategory::UseAsStatic
} else {
ConstraintCategory::UseAsConst
}
} else {
ConstraintCategory::Return(ReturnConstraint::Normal)
}
}
Some(l)
if matches!(body.local_decls[l].local_info(), LocalInfo::AggregateTemp) =>
{
ConstraintCategory::Usage
}
Some(l) if !body.local_decls[l].is_user_variable() => {
ConstraintCategory::Boring
}
_ => ConstraintCategory::Assignment,
};

E.g. by checking for RefForGuard and looking at VarBindingForm::opt_match_place. It's a bit fiddly sadly

Copy link
Contributor

Choose a reason for hiding this comment

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

One likely quite impactful suggestion: make it boring if opt_ty_info is None

Copy link
Contributor Author

Choose a reason for hiding this comment

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

that's where I was looking, yeah. though I was focusing too much on distinguishing between assignment syntaxes. opt_match_place does work for catching those by-ref bindings. and since we (probably?) don't need the span or place from that, checking the VarBindingForm's binding_mode should be equivalent (I think). I don't think they should all be boring though, so there'd at need to be some other check (if not a different check entirely). e.g., for field-projection-mutating-context.rs:

1	error[E0308]: mismatched types
-	   --> $DIR/field-projection-mutating-context.rs:9:13
+	   --> $DIR/field-projection-mutating-context.rs:9:25
3	    |
4	 LL |     let Foo(ref mut y): Foo<fn(&'static str)> = x;
-	    |             ^^^^^^^^^ one type is more general than the other
+	    |                         ^^^^^^^^^^^^^^^^^^^^^ one type is more general than the other
6	    |
7	    = note: expected fn pointer `for<'a> fn(&'a _)`
8	               found fn pointer `fn(&_)`

the assignment is what relates the two types, so it makes more sense to blame than just one of the type annotations. there's also fn-subtype.stderr specifically checking for an assignment being blamed, though that assignment is by-value. in both cases the span associated with the assignment's constraint is unfortunate, but it is the assignment being blamed.

RefForGuard locals being boring to assign to makes sense, but it doesn't look like we blame those assignments in any tests anyway. I'd have to think more to come up with an example, if it's possible.

and opt_ty_info isn't quite what the docs claim. it looks like it's only Some(_) for function inputs (specifically for the inputs themselves, not projections from destructuring patterns). any other binding gets an opt_ty_info of None even if there's an annotation on it. we can get user annotations on normal local variables from a local decl's user_ty field, though. there's a slight asymmetry in how UserTypeProjections accounts nicely for projections and opt_ty_info doesn't, but also I can't imagine ever wanting to point to constraints from an argument being destructured, so that's probably fine (if unsettlingly ad-hoc). just checking for those also misses assignments from paths with explicitly-provided generic arguments, but maybe that's fine too; I think getting all user types and sorting out what's inferred from what's provided would be a much too involved process. in most of the tests this affects, it doesn't matter too much whether the assignment or something else gets blamed; they each provide half of the reason for a region error, so unless we surface more constraints' spans, the user has to hunt down the other half themself either way.

@bors
Copy link
Contributor

bors commented Dec 18, 2024

☔ The latest upstream changes (presumably #134243) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the better-blame-constraints-for-static branch from 326a2bc to b041c50 Compare December 19, 2024 09:35
@dianne
Copy link
Contributor Author

dianne commented Dec 19, 2024

I've rebased and added an experiment in treating some assignments without sufficiently nearby type annotations as uninteresting. this reverts several tests' output to what they were prior to this PR.

@bors
Copy link
Contributor

bors commented Jan 7, 2025

✌️ @dianne, you can now approve this pull request!

If @lcnr told you to "r=me" after making some further change, please make that change, then do @bors r=@lcnr

@dianne
Copy link
Contributor Author

dianne commented Jan 7, 2025

I've looked into those. I'll leave individual replies for posterity/reference, but in short I think they'd either require bigger or unrelated changes, so I'm inclined to leave them for future work. thanks again!

@bors r=@lcnr

@bors
Copy link
Contributor

bors commented Jan 7, 2025

📌 Commit fe8b12f has been approved by lcnr

It is now in the queue for this repository.

@bors bors added S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Jan 7, 2025
@bors
Copy link
Contributor

bors commented Jan 8, 2025

⌛ Testing commit fe8b12f with merge 6afee11...

@bors
Copy link
Contributor

bors commented Jan 8, 2025

☀️ Test successful - checks-actions
Approved by: lcnr
Pushing 6afee11 to master...

@bors bors added the merged-by-bors This PR was explicitly merged by bors. label Jan 8, 2025
@bors bors merged commit 6afee11 into rust-lang:master Jan 8, 2025
7 checks passed
@rustbot rustbot added this to the 1.86.0 milestone Jan 8, 2025
@rust-timer
Copy link
Collaborator

Finished benchmarking commit (6afee11): comparison URL.

Overall result: ❌✅ regressions and improvements - please read the text below

Our benchmarks found a performance regression caused by this PR.
This might be an actual regression, but it can also be just noise.

Next Steps:

  • If the regression was expected or you think it can be justified,
    please write a comment with sufficient written justification, and add
    @rustbot label: +perf-regression-triaged to it, to mark the regression as triaged.
  • If you think that you know of a way to resolve the regression, try to create
    a new PR with a fix for the regression.
  • If you do not understand the regression or you think that it is just noise,
    you can ask the @rust-lang/wg-compiler-performance working group for help (members of this group
    were already notified of this PR).

@rustbot label: +perf-regression
cc @rust-lang/wg-compiler-performance

Instruction count

This is the most reliable metric that we have; it was used to determine the overall result at the top of this comment. However, even this metric can sometimes exhibit noise.

mean range count
Regressions ❌
(primary)
3.2% [0.1%, 21.0%] 17
Regressions ❌
(secondary)
- - 0
Improvements ✅
(primary)
- - 0
Improvements ✅
(secondary)
-0.3% [-0.3%, -0.3%] 1
All ❌✅ (primary) 3.2% [0.1%, 21.0%] 17

Max RSS (memory usage)

Results (primary 2.7%, secondary -1.8%)

This is a less reliable metric that may be of interest but was not used to determine the overall result at the top of this comment.

mean range count
Regressions ❌
(primary)
4.3% [2.5%, 6.1%] 3
Regressions ❌
(secondary)
- - 0
Improvements ✅
(primary)
-2.1% [-2.1%, -2.1%] 1
Improvements ✅
(secondary)
-1.8% [-2.1%, -1.4%] 2
All ❌✅ (primary) 2.7% [-2.1%, 6.1%] 4

Cycles

Results (primary 4.7%, secondary -1.9%)

This is a less reliable metric that may be of interest but was not used to determine the overall result at the top of this comment.

mean range count
Regressions ❌
(primary)
4.7% [0.8%, 20.1%] 14
Regressions ❌
(secondary)
2.9% [2.1%, 3.9%] 5
Improvements ✅
(primary)
- - 0
Improvements ✅
(secondary)
-3.2% [-4.7%, -1.4%] 17
All ❌✅ (primary) 4.7% [0.8%, 20.1%] 14

Binary size

This benchmark run did not return any relevant results for this metric.

Bootstrap: 764.901s -> 763.722s (-0.15%)
Artifact size: 325.67 MiB -> 325.68 MiB (0.01%)

@rustbot rustbot added the perf-regression Performance regression. label Jan 8, 2025
@Kobzol
Copy link
Contributor

Kobzol commented Jan 8, 2025

Oh, that perf. regression looks quite bad! Probably it was introduced after some later changes, because a previous perf. run did not show any regressions. Looks like some MIR queries were executed more often.

If the fix is obvious, it would be great to send a PR with a fix. If not, I'd suggest reverting this first, since the regression seems to affect real world crates.

@lqd
Copy link
Member

lqd commented Jan 8, 2025

Unless these benchmarks emit borrowck errors (unlikely), the regressions look real. There may be additional queries on the happy path.

image

lqd added a commit to lqd/rust that referenced this pull request Jan 8, 2025
…nts-for-static, r=lcnr"

This reverts commit 6afee11, reversing
changes made to 9c87288.
@lqd
Copy link
Member

lqd commented Jan 8, 2025

I've prepared a revert in #135263 that we can land in case the fix is not obvious or @dianne doesn't have time to take a look.

lqd added a commit to lqd/rust that referenced this pull request Jan 8, 2025
…nts-for-static, r=lcnr"

This reverts commit 6afee11, reversing
changes made to 9c87288.
@dianne
Copy link
Contributor Author

dianne commented Jan 9, 2025

I have an idea of what it could be.. kind of. I'm not confident, and I can't seem to get the profiler running locally to be sure, but I've opened a PR: #135273.

Since the regressions come up during incremental compilation and seem to be query-system-related, my guess is that I committed a query sin, resulting in lots, lots more executions of mir_borrowck. I made a tweak to MIR typeck to treat certain MIR assignments that weren't written by the user as not written by the user. In order to avoid adding more information to the MIR or tracking more things during pattern lowering, I wrote a bit of a hack that queried the HIR. If that required computing something not in the incremental cache, or if it somehow clobbered/invalidated it, that could explain the huge number of additional queries. I'm not too familiar with the query system or incremental compilation, so I could be way off base.

In case that isn't the fix, I'd say go ahead with the revert. My other ideas for what could be causing the regression aren't too solid and would require more looking-into. For personal reference (or if anyone can help), those ideas are:

  • best_blame_constraint gets called on the happy path in order to assign a category and span to region constraints propagated from checking closure bodies. This PR changes the categories and spans returned from that, but I'm not sure how it'd result in all those additional queries.
  • Maybe something I thought was only for diagnostics gets called on the happy path too. I don't know what it'd be, how it'd get called, or how it'd result in those queries, though.

bors added a commit to rust-lang-ci/rust that referenced this pull request Jan 9, 2025
…ing, r=<try>

Remove special-casing for argument patterns in MIR typeck (attempt to fix perf regression of  rust-lang#133858)

See [my comment](rust-lang#133858 (comment)) on rust-lang#133858 for more information. This is just a guess as to what went wrong, and I haven't been able to get the profiler running locally, so I'll need a perf run to make sure this actually helps.

There's one test's stderr that suffers a bit, but this was just papering over the issue anyway. Making region errors point to the correct constraints in the presence of invariance/contravariance is a broader problem; the current way it's handled is mostly based on guesswork, luck, and hoping it works out. Properly handling that (somehow) would improve the test's stderr without the hack that this PR reverts.
bors added a commit to rust-lang-ci/rust that referenced this pull request Jan 9, 2025
…ing, r=lqd

Remove special-casing for argument patterns in MIR typeck (attempt to fix perf regression of  rust-lang#133858)

See [my comment](rust-lang#133858 (comment)) on rust-lang#133858 for more information. This is just a guess as to what went wrong, and I haven't been able to get the profiler running locally, so I'll need a perf run to make sure this actually helps.

There's one test's stderr that suffers a bit, but this was just papering over the issue anyway. Making region errors point to the correct constraints in the presence of invariance/contravariance is a broader problem; the current way it's handled is mostly based on guesswork, luck, and hoping it works out. Properly handling that (somehow) would improve the test's stderr without the hack that this PR reverts.
bors added a commit to rust-lang-ci/rust that referenced this pull request Jan 10, 2025
…ing, r=lqd

Remove special-casing for argument patterns in MIR typeck (attempt to fix perf regression of  rust-lang#133858)

See [my comment](rust-lang#133858 (comment)) on rust-lang#133858 for more information. This is just a guess as to what went wrong, and I haven't been able to get the profiler running locally, so I'll need a perf run to make sure this actually helps.

There's one test's stderr that suffers a bit, but this was just papering over the issue anyway. Making region errors point to the correct constraints in the presence of invariance/contravariance is a broader problem; the current way it's handled is mostly based on guesswork, luck, and hoping it works out. Properly handling that (somehow) would improve the test's stderr without the hack that this PR reverts.
@lqd
Copy link
Member

lqd commented Jan 10, 2025

The perf regression was fixed in #135273 (comment)
marking as triaged @rustbot label: +perf-regression-triaged

@rustbot rustbot added the perf-regression-triaged The performance regression has been triaged. label Jan 10, 2025
github-merge-queue bot pushed a commit to rust-lang/rust-analyzer that referenced this pull request Jan 20, 2025
Remove special-casing for argument patterns in MIR typeck (attempt to fix perf regression of  #133858)

See [my comment](rust-lang/rust#133858 (comment)) on #133858 for more information. This is just a guess as to what went wrong, and I haven't been able to get the profiler running locally, so I'll need a perf run to make sure this actually helps.

There's one test's stderr that suffers a bit, but this was just papering over the issue anyway. Making region errors point to the correct constraints in the presence of invariance/contravariance is a broader problem; the current way it's handled is mostly based on guesswork, luck, and hoping it works out. Properly handling that (somehow) would improve the test's stderr without the hack that this PR reverts.
Kobzol pushed a commit to Kobzol/rustc-dev-guide that referenced this pull request Jan 20, 2025
Remove special-casing for argument patterns in MIR typeck (attempt to fix perf regression of  #133858)

See [my comment](rust-lang/rust#133858 (comment)) on #133858 for more information. This is just a guess as to what went wrong, and I haven't been able to get the profiler running locally, so I'll need a perf run to make sure this actually helps.

There's one test's stderr that suffers a bit, but this was just papering over the issue anyway. Making region errors point to the correct constraints in the presence of invariance/contravariance is a broader problem; the current way it's handled is mostly based on guesswork, luck, and hoping it works out. Properly handling that (somehow) would improve the test's stderr without the hack that this PR reverts.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
merged-by-bors This PR was explicitly merged by bors. perf-regression Performance regression. perf-regression-triaged The performance regression has been triaged. S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Confusing error message for lifetime mismatch
9 participants