Rollup merge of #63833 - estebank:suggest-closure-call, r=petrochenkov
Suggest calling closure with resolved return type when appropriate Follow up to #63337. CC #63100. ``` error[E0308]: mismatched types --> $DIR/fn-or-tuple-struct-without-args.rs:46:20 | LL | let closure = || 42; | -- closure defined here LL | let _: usize = closure; | ^^^^^^^ | | | expected usize, found closure | help: use parentheses to call this closure: `closure()` | = note: expected type `usize` found type `[closure@$DIR/fn-or-tuple-struct-without-args.rs:45:19: 45:24]` ```
This commit is contained in:
commit
3d4b1135f8
@ -385,7 +385,7 @@ impl<'tcx> ClosureSubsts<'tcx> {
|
||||
let ty = self.closure_sig_ty(def_id, tcx);
|
||||
match ty.sty {
|
||||
ty::FnPtr(sig) => sig,
|
||||
_ => bug!("closure_sig_ty is not a fn-ptr: {:?}", ty),
|
||||
_ => bug!("closure_sig_ty is not a fn-ptr: {:?}", ty.sty),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -799,12 +799,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||
/// adjusted type of the expression, if successful.
|
||||
/// Adjustments are only recorded if the coercion succeeded.
|
||||
/// The expressions *must not* have any pre-existing adjustments.
|
||||
pub fn try_coerce(&self,
|
||||
pub fn try_coerce(
|
||||
&self,
|
||||
expr: &hir::Expr,
|
||||
expr_ty: Ty<'tcx>,
|
||||
target: Ty<'tcx>,
|
||||
allow_two_phase: AllowTwoPhase)
|
||||
-> RelateResult<'tcx, Ty<'tcx>> {
|
||||
allow_two_phase: AllowTwoPhase,
|
||||
) -> RelateResult<'tcx, Ty<'tcx>> {
|
||||
let source = self.resolve_type_vars_with_obligations(expr_ty);
|
||||
debug!("coercion::try({:?}: {:?} -> {:?})", expr, source, target);
|
||||
|
||||
|
@ -3917,25 +3917,32 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||
expected: Ty<'tcx>,
|
||||
found: Ty<'tcx>,
|
||||
) -> bool {
|
||||
match found.sty {
|
||||
ty::FnDef(..) | ty::FnPtr(_) => {}
|
||||
_ => return false,
|
||||
}
|
||||
let hir = self.tcx.hir();
|
||||
let (def_id, sig) = match found.sty {
|
||||
ty::FnDef(def_id, _) => (def_id, found.fn_sig(self.tcx)),
|
||||
ty::Closure(def_id, substs) => {
|
||||
// We don't use `closure_sig` to account for malformed closures like
|
||||
// `|_: [_; continue]| {}` and instead we don't suggest anything.
|
||||
let closure_sig_ty = substs.closure_sig_ty(def_id, self.tcx);
|
||||
(def_id, match closure_sig_ty.sty {
|
||||
ty::FnPtr(sig) => sig,
|
||||
_ => return false,
|
||||
})
|
||||
}
|
||||
_ => return false,
|
||||
};
|
||||
|
||||
let sig = found.fn_sig(self.tcx);
|
||||
let sig = self
|
||||
.replace_bound_vars_with_fresh_vars(expr.span, infer::FnCall, &sig)
|
||||
.0;
|
||||
let sig = self.normalize_associated_types_in(expr.span, &sig);
|
||||
if let Ok(_) = self.try_coerce(expr, sig.output(), expected, AllowTwoPhase::No) {
|
||||
if self.can_coerce(sig.output(), expected) {
|
||||
let (mut sugg_call, applicability) = if sig.inputs().is_empty() {
|
||||
(String::new(), Applicability::MachineApplicable)
|
||||
} else {
|
||||
("...".to_string(), Applicability::HasPlaceholders)
|
||||
};
|
||||
let mut msg = "call this function";
|
||||
if let ty::FnDef(def_id, ..) = found.sty {
|
||||
match hir.get_if_local(def_id) {
|
||||
Some(Node::Item(hir::Item {
|
||||
node: ItemKind::Fn(.., body_id),
|
||||
@ -3957,6 +3964,24 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||
_ => "_".to_string(),
|
||||
}).collect::<Vec<_>>().join(", ");
|
||||
}
|
||||
Some(Node::Expr(hir::Expr {
|
||||
node: ExprKind::Closure(_, _, body_id, closure_span, _),
|
||||
span: full_closure_span,
|
||||
..
|
||||
})) => {
|
||||
if *full_closure_span == expr.span {
|
||||
return false;
|
||||
}
|
||||
err.span_label(*closure_span, "closure defined here");
|
||||
msg = "call this closure";
|
||||
let body = hir.body(*body_id);
|
||||
sugg_call = body.arguments.iter()
|
||||
.map(|arg| match &arg.pat.node {
|
||||
hir::PatKind::Binding(_, _, ident, None)
|
||||
if ident.name != kw::SelfLower => ident.to_string(),
|
||||
_ => "_".to_string(),
|
||||
}).collect::<Vec<_>>().join(", ");
|
||||
}
|
||||
Some(Node::Ctor(hir::VariantData::Tuple(fields, _))) => {
|
||||
sugg_call = fields.iter().map(|_| "_").collect::<Vec<_>>().join(", ");
|
||||
match hir.as_local_hir_id(def_id).and_then(|hir_id| hir.def_kind(hir_id)) {
|
||||
@ -3985,7 +4010,6 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||
.join(", "),
|
||||
_ => {}
|
||||
}
|
||||
};
|
||||
if let Ok(code) = self.sess().source_map().span_to_snippet(expr.span) {
|
||||
err.span_suggestion(
|
||||
expr.span,
|
||||
|
@ -42,4 +42,6 @@ fn main() {
|
||||
let _: usize = X::bal; //~ ERROR mismatched types
|
||||
let _: usize = X.ban; //~ ERROR attempted to take value of method
|
||||
let _: usize = X.bal; //~ ERROR attempted to take value of method
|
||||
let closure = || 42;
|
||||
let _: usize = closure; //~ ERROR mismatched types
|
||||
}
|
||||
|
@ -214,7 +214,21 @@ error[E0615]: attempted to take value of method `bal` on type `X`
|
||||
LL | let _: usize = X.bal;
|
||||
| ^^^ help: use parentheses to call the method: `bal()`
|
||||
|
||||
error: aborting due to 16 previous errors
|
||||
error[E0308]: mismatched types
|
||||
--> $DIR/fn-or-tuple-struct-without-args.rs:46:20
|
||||
|
|
||||
LL | let closure = || 42;
|
||||
| -- closure defined here
|
||||
LL | let _: usize = closure;
|
||||
| ^^^^^^^
|
||||
| |
|
||||
| expected usize, found closure
|
||||
| help: use parentheses to call this closure: `closure()`
|
||||
|
|
||||
= note: expected type `usize`
|
||||
found type `[closure@$DIR/fn-or-tuple-struct-without-args.rs:45:19: 45:24]`
|
||||
|
||||
error: aborting due to 17 previous errors
|
||||
|
||||
Some errors have detailed explanations: E0308, E0423, E0615.
|
||||
For more information about an error, try `rustc --explain E0308`.
|
||||
|
Loading…
Reference in New Issue
Block a user