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

Implement suggestions for elided_named_lifetimes #129840

Merged

Conversation

GrigorenkoPV
Copy link
Contributor

@GrigorenkoPV GrigorenkoPV commented Sep 1, 2024

A follow-up to #129207, as per #129207 (comment).

r? cjgillot

I will probably squash this a bit, but later.

@rustbot label +A-lint

@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. T-libs Relevant to the library team, which will review and decide on the PR/issue. A-lints Area: Lints (warnings about flaws in source code) such as unused_mut. and removed T-libs Relevant to the library team, which will review and decide on the PR/issue. labels Sep 1, 2024
@rust-log-analyzer

This comment has been minimized.

@GrigorenkoPV GrigorenkoPV force-pushed the elided-named-lifetimes-suggestion branch from 2b60ebc to 75bb732 Compare September 1, 2024 13:50
@GrigorenkoPV
Copy link
Contributor Author

GrigorenkoPV commented Sep 1, 2024

Apparently this is in conflict with clippy's needless_lifetimes. For

fn foo<'a>(x: &'a str) -> &str

this suggests adding lifetime to the output, but clippy suggests removing it from the input. Applying both at the same time produces

fn foo(x: &str) -> &'a str

which is not correct.

I would say that needless_lifetimes here is probably a more sensible suggestion than elided_named_lifetimes.

Though, if one applies elided_named_lifetimes first, then needless_lifetimes is still able to tell that the lifetime can be removed (now from both input and output), and the result is not broken.

Idk how to fix this one. cc @rust-lang/clippy

@GrigorenkoPV GrigorenkoPV marked this pull request as ready for review September 1, 2024 13:51
@rustbot
Copy link
Collaborator

rustbot commented Sep 1, 2024

Some changes occurred in src/tools/clippy

cc @rust-lang/clippy

@rust-log-analyzer

This comment has been minimized.

@GrigorenkoPV GrigorenkoPV force-pushed the elided-named-lifetimes-suggestion branch from 75bb732 to 9e2d264 Compare September 6, 2024 14:06
@GrigorenkoPV
Copy link
Contributor Author

GrigorenkoPV commented Sep 6, 2024

It seems like without suggestions, elided_named_lifetimes can be quite confusing: #129207 (comment), so I feel like it is a priority to get this PR out in a timely manner, but there is a blocker at the moment: #129840 (comment)

I see three possible ways of dealing with the issue.

  1. Hack around the conflict now

    This is what I've done: 9e2d264. This commit simply disables the suggestions for non-'static cases1.

    Pros: no stalling, adds some suggestions, there is no conflict, the hack is easy to revert once we resolve the issue.
    Cons: many suggestions are missing.

  2. Ignore the conflict

    Just merge as is (removing the failing test), and deal with it later. The fallout of the conflict would probably be rather small2.

    Pros: no stalling, all suggestions are added.
    Cons: potential conflicts with a clippy complexity lint.

  3. Resolve this now

    Do not merge this PR until the issue is resolved.

    Pros: all suggestions are added, there is no conflict
    Cons: high risk of missing the 1.83 release window, which elided_named_lifetimes has landed into.

Footnotes

  1. I've tried setting applicability for potentially conflicting cases to all other variants instead of MachineApplicable, but it didn't help.

  2. citation needed

@cjgillot
Copy link
Contributor

cjgillot commented Sep 6, 2024

@bors r+ rollup

The easiest way to remove the conflict could be to uplift the needless_lifetimes lint to rustc. It may fit well with the current architecture of the code, but I haven't tried writing any code.

@bors
Copy link
Contributor

bors commented Sep 6, 2024

📌 Commit 9e2d264 has been approved by cjgillot

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 Sep 6, 2024
@GrigorenkoPV
Copy link
Contributor Author

GrigorenkoPV commented Sep 7, 2024

Okay, I've managed to swap the priorities between elided_named_lifetimes and cippy::needless_lifetimes. Now the latter does not emit when the former would.1

Patch
diff --git a/compiler/rustc_lint/src/lints.rs b/compiler/rustc_lint/src/lints.rs
index ae7e9659856..693f1b22587 100644
--- a/compiler/rustc_lint/src/lints.rs
+++ b/compiler/rustc_lint/src/lints.rs
@@ -2639,13 +2639,6 @@ fn decorate_lint(self, diag: &mut rustc_errors::Diag<'_, G>) {
         if let Some(declaration) = declaration {
             diag.span_label(declaration, fluent::lint_label_named);
         }
-        // FIXME(GrigorenkoPV): this `if` and `return` should be removed,
-        //  but currently this lint's suggestions can conflict with those of `clippy::needless_lifetimes`:
-        //  https://github.com/rust-lang/rust/pull/129840#issuecomment-2323349119
-        // HACK: `'static` suggestions will never sonflict, emit only those for now.
-        if name != rustc_span::symbol::kw::StaticLifetime {
-            return;
-        }
         match kind {
             MissingLifetimeKind::Underscore => diag.span_suggestion_verbose(
                 span,
diff --git a/src/tools/clippy/clippy_lints/src/lifetimes.rs b/src/tools/clippy/clippy_lints/src/lifetimes.rs
index ba34af9c100..793ef442c13 100644
--- a/src/tools/clippy/clippy_lints/src/lifetimes.rs
+++ b/src/tools/clippy/clippy_lints/src/lifetimes.rs
@@ -420,6 +420,9 @@ fn could_use_elision<'tcx>(
         .filter_map(|(def_id, occurrences)| {
             if occurrences == 1
                 && (input_lts.len() == 1 || !output_lts.iter().any(|lt| named_lifetime(lt) == Some(def_id)))
+                && !output_lts
+                    .iter()
+                    .any(|lt| suggested_by_elided_named_lifetimes(lt, def_id))
             {
                 Some(def_id)
             } else {
@@ -437,6 +440,10 @@ fn could_use_elision<'tcx>(
     Some((elidable_lts, usages))
 }
 
+fn suggested_by_elided_named_lifetimes(lt: &Lifetime, res: LocalDefId) -> bool {
+    lt.res == LifetimeName::Param(res) && matches!(lt.ident.name, kw::Empty | kw::UnderscoreLifetime)
+}
+
 fn allowed_lts_from(named_generics: &[GenericParam<'_>]) -> FxHashSet<LocalDefId> {
     named_generics
         .iter()

but the problem now is that

if one applies elided_named_lifetimes first, then needless_lifetimes is still able to tell that the lifetime can be removed (now from both input and output)

This does result in only correct code along the way, but it makes a new suggestion pop up after the previous is applied. The testsuite does not like that:

error: test got exit status: 1, but expected 0
   --> tests/ui/needless_lifetimes.fixed:266:30
    |
266 | fn named_input_elided_output<'a>(_arg: &'a str) -> &'a str {
    |                              ^^ after rustfix is applied, all errors should be gone, but weren't
    |

So, to reiterate my previous point:

I would say that needless_lifetimes here is probably a more sensible suggestion than elided_named_lifetimes.

We should probably emit needless_lifetimes first, cache the set of lifetime params that were suggested to be elided, and then just avoid those in elided_named_lifetimes.

The problem is that we emit elided_named_lifetimes immediately during lifetime resolution, while needless_lifetimes walks the tree after the fact.

I don't think we can move needless_lifetimes inside the resolution (it fundamentally works with already resolved lifetimes and walks the tree quite a lot). But maybe we can move elided_named_lifetimes outside of resolution and into the same pass as needless_lifetimes.2

The easiest way to remove the conflict could be to uplift the needless_lifetimes lint to rustc.

It seems like uplifting is unavoidable here, because we need some information to flow from needless_lifetimes to elided_named_lifetimes. But needless_lifetimes currently has some issues:

There was an attempt to fix those but it seemingly got abandoned right at the start: rust-lang/rust-clippy#13141 rust-lang/rust-clippy#12456

Soo, should we maybe just break needless_lifetimes even further and go with the option 2?

Footnotes

  1. And unlike with the 9e2d264 hack, the suggestion is suppressed only when there would be a conflict, not when there potentially could be a conflict.

  2. It seems like we can easily detect the lifetimes that needless_lifetimes is interested in using matches!(lt.res, LifetimeName::Param(_) | LifetimeName::Static) && matches!(lt.ident.name, kw::Empty | kw::UnderscoreLifetime). Though, we do lose access to MissingLifetimeKind, which was quite handy for suggestions. But maybe we can propagate it or infer it from the context. Speaking of inferring from the context, moving elided_named_lifetimes to a LateLintPass will probably eliminate the need for the LifetimeRes::Static.suppress_elision_warning & MissingLifetime.id_for_lint hacks.

bors added a commit to rust-lang-ci/rust that referenced this pull request Sep 7, 2024
…mpiler-errors

Rollup of 9 pull requests

Successful merges:

 - rust-lang#128871 (bypass linker configuration and cross target check for specific commands)
 - rust-lang#129468 ([testsuite][cleanup] Remove all usages of `dont_merge` hack to avoid function merging)
 - rust-lang#129614 (Adjust doc comment of Condvar::wait_while)
 - rust-lang#129840 (Implement suggestions for `elided_named_lifetimes`)
 - rust-lang#129891 (Do not request sanitizers for naked functions)
 - rust-lang#129899 (Add Suggestions for Misspelled Keywords)
 - rust-lang#129940 (s390x: Fix a regression related to backchain feature)
 - rust-lang#129987 (Don't store region in `CapturedPlace`)
 - rust-lang#130054 (Add missing quotation marks)

r? `@ghost`
`@rustbot` modify labels: rollup
bors added a commit to rust-lang-ci/rust that referenced this pull request Sep 7, 2024
…mpiler-errors

Rollup of 9 pull requests

Successful merges:

 - rust-lang#128871 (bypass linker configuration and cross target check for specific commands)
 - rust-lang#129468 ([testsuite][cleanup] Remove all usages of `dont_merge` hack to avoid function merging)
 - rust-lang#129614 (Adjust doc comment of Condvar::wait_while)
 - rust-lang#129840 (Implement suggestions for `elided_named_lifetimes`)
 - rust-lang#129891 (Do not request sanitizers for naked functions)
 - rust-lang#129899 (Add Suggestions for Misspelled Keywords)
 - rust-lang#129940 (s390x: Fix a regression related to backchain feature)
 - rust-lang#129987 (Don't store region in `CapturedPlace`)
 - rust-lang#130054 (Add missing quotation marks)

r? `@ghost`
`@rustbot` modify labels: rollup
@bors bors merged commit 41a20dc into rust-lang:master Sep 7, 2024
6 checks passed
@rustbot rustbot added this to the 1.83.0 milestone Sep 7, 2024
rust-timer added a commit to rust-lang-ci/rust that referenced this pull request Sep 7, 2024
Rollup merge of rust-lang#129840 - GrigorenkoPV:elided-named-lifetimes-suggestion, r=cjgillot

Implement suggestions for `elided_named_lifetimes`

A follow-up to rust-lang#129207, as per rust-lang#129207 (comment).

r? cjgillot

I will probably squash this a bit, but later.

`@rustbot` label +A-lint
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-lints Area: Lints (warnings about flaws in source code) such as unused_mut. 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.

5 participants