Skip to content

Commit c3eb62a

Browse files
authored
Unrolled build for rust-lang#125684
Rollup merge of rust-lang#125684 - estebank:pin-to-binding-suggestion, r=pnkfelix Account for existing bindings when suggesting `pin!()` When we encounter a situation where we'd suggest `pin!()`, we now account for that expression existing as part of an assignment and provide an appropriate suggestion: ``` error[E0599]: no method named `poll` found for type parameter `F` in the current scope --> $DIR/pin-needed-to-poll-3.rs:19:28 | LL | impl<F> Future for FutureWrapper<F> | - method `poll` not found for this type parameter ... LL | let res = self.fut.poll(cx); | ^^^^ method not found in `F` | help: consider pinning the expression | LL ~ let mut pinned = std::pin::pin!(self.fut); LL ~ let res = pinned.as_mut().poll(cx); | ``` Fix rust-lang#125661.
2 parents d0227c6 + 5585f31 commit c3eb62a

File tree

3 files changed

+103
-8
lines changed

3 files changed

+103
-8
lines changed

compiler/rustc_hir_typeck/src/method/suggest.rs

+60-8
Original file line numberDiff line numberDiff line change
@@ -3360,14 +3360,66 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
33603360
.source_map()
33613361
.indentation_before(rcvr.span)
33623362
.unwrap_or_else(|| " ".to_string());
3363-
err.multipart_suggestion(
3364-
"consider pinning the expression",
3365-
vec![
3366-
(rcvr.span.shrink_to_lo(), format!("let mut pinned = std::pin::pin!(")),
3367-
(rcvr.span.shrink_to_hi(), format!(");\n{indent}pinned.{pin_call}()")),
3368-
],
3369-
Applicability::MaybeIncorrect,
3370-
);
3363+
let mut expr = rcvr;
3364+
while let Node::Expr(call_expr) = self.tcx.parent_hir_node(expr.hir_id)
3365+
&& let hir::ExprKind::MethodCall(hir::PathSegment { .. }, ..) =
3366+
call_expr.kind
3367+
{
3368+
expr = call_expr;
3369+
}
3370+
match self.tcx.parent_hir_node(expr.hir_id) {
3371+
Node::LetStmt(stmt)
3372+
if let Some(init) = stmt.init
3373+
&& let Ok(code) =
3374+
self.tcx.sess.source_map().span_to_snippet(rcvr.span) =>
3375+
{
3376+
// We need to take care to account for the existing binding when we
3377+
// suggest the code.
3378+
err.multipart_suggestion(
3379+
"consider pinning the expression",
3380+
vec![
3381+
(
3382+
stmt.span.shrink_to_lo(),
3383+
format!(
3384+
"let mut pinned = std::pin::pin!({code});\n{indent}"
3385+
),
3386+
),
3387+
(
3388+
init.span.until(rcvr.span.shrink_to_hi()),
3389+
format!("pinned.{pin_call}()"),
3390+
),
3391+
],
3392+
Applicability::MaybeIncorrect,
3393+
);
3394+
}
3395+
Node::Block(_) | Node::Stmt(_) => {
3396+
// There's no binding, so we can provide a slightly nicer looking
3397+
// suggestion.
3398+
err.multipart_suggestion(
3399+
"consider pinning the expression",
3400+
vec![
3401+
(
3402+
rcvr.span.shrink_to_lo(),
3403+
format!("let mut pinned = std::pin::pin!("),
3404+
),
3405+
(
3406+
rcvr.span.shrink_to_hi(),
3407+
format!(");\n{indent}pinned.{pin_call}()"),
3408+
),
3409+
],
3410+
Applicability::MaybeIncorrect,
3411+
);
3412+
}
3413+
_ => {
3414+
// We don't quite know what the users' code looks like, so we don't
3415+
// provide a pinning suggestion.
3416+
err.span_help(
3417+
rcvr.span,
3418+
"consider pinning the expression with `std::pin::pin!()` and \
3419+
assigning that to a new binding",
3420+
);
3421+
}
3422+
}
33713423
// We don't care about the other suggestions.
33723424
alt_rcvr_sugg = true;
33733425
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
use std::{
2+
future::Future,
3+
pin::Pin,
4+
task::{Context, Poll},
5+
};
6+
7+
8+
struct FutureWrapper<F> {
9+
fut: F,
10+
}
11+
12+
impl<F> Future for FutureWrapper<F>
13+
where
14+
F: Future,
15+
{
16+
type Output = F::Output;
17+
18+
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
19+
let res = self.fut.poll(cx);
20+
//~^ ERROR no method named `poll` found for type parameter `F` in the current scope
21+
res
22+
}
23+
}
24+
25+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
error[E0599]: no method named `poll` found for type parameter `F` in the current scope
2+
--> $DIR/pin-needed-to-poll-3.rs:19:28
3+
|
4+
LL | impl<F> Future for FutureWrapper<F>
5+
| - method `poll` not found for this type parameter
6+
...
7+
LL | let res = self.fut.poll(cx);
8+
| ^^^^ method not found in `F`
9+
|
10+
help: consider pinning the expression
11+
|
12+
LL ~ let mut pinned = std::pin::pin!(self.fut);
13+
LL ~ let res = pinned.as_mut().poll(cx);
14+
|
15+
16+
error: aborting due to 1 previous error
17+
18+
For more information about this error, try `rustc --explain E0599`.

0 commit comments

Comments
 (0)