Rollup merge of #132489 - compiler-errors:fn-sugg-tweaks, r=BoxyUwU
Fix closure arg extraction in `extract_callable_info`, generalize it to async closures * Fix argument extraction in `extract_callable_info` * FIx `extract_callable_info` to work for async closures * Remove redundant `is_fn_ty` which is just a less general `extract_callable_info` * More precisely name what is being called (i.e. call it a "closure" not a "function") Review this without whitespace -- I ended up reformatting `extract_callable_info` because some pesky `//` comments were keeping the let-chains from being formatted.
This commit is contained in:
commit
c064f6e1fc
9 changed files with 148 additions and 139 deletions
|
@ -31,6 +31,7 @@ use rustc_span::symbol::{Ident, kw, sym};
|
||||||
use rustc_span::{
|
use rustc_span::{
|
||||||
DUMMY_SP, ErrorGuaranteed, ExpnKind, FileName, MacroKind, Span, Symbol, edit_distance,
|
DUMMY_SP, ErrorGuaranteed, ExpnKind, FileName, MacroKind, Span, Symbol, edit_distance,
|
||||||
};
|
};
|
||||||
|
use rustc_trait_selection::error_reporting::traits::DefIdOrName;
|
||||||
use rustc_trait_selection::error_reporting::traits::on_unimplemented::OnUnimplementedNote;
|
use rustc_trait_selection::error_reporting::traits::on_unimplemented::OnUnimplementedNote;
|
||||||
use rustc_trait_selection::infer::InferCtxtExt;
|
use rustc_trait_selection::infer::InferCtxtExt;
|
||||||
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _;
|
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _;
|
||||||
|
@ -45,50 +46,6 @@ use crate::errors::{self, CandidateTraitNote, NoAssociatedItem};
|
||||||
use crate::{Expectation, FnCtxt};
|
use crate::{Expectation, FnCtxt};
|
||||||
|
|
||||||
impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
fn is_fn_ty(&self, ty: Ty<'tcx>, span: Span) -> bool {
|
|
||||||
let tcx = self.tcx;
|
|
||||||
match ty.kind() {
|
|
||||||
// Not all of these (e.g., unsafe fns) implement `FnOnce`,
|
|
||||||
// so we look for these beforehand.
|
|
||||||
// FIXME(async_closures): These don't impl `FnOnce` by default.
|
|
||||||
ty::Closure(..) | ty::FnDef(..) | ty::FnPtr(..) => true,
|
|
||||||
// If it's not a simple function, look for things which implement `FnOnce`.
|
|
||||||
_ => {
|
|
||||||
let Some(fn_once) = tcx.lang_items().fn_once_trait() else {
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
// This conditional prevents us from asking to call errors and unresolved types.
|
|
||||||
// It might seem that we can use `predicate_must_hold_modulo_regions`,
|
|
||||||
// but since a Dummy binder is used to fill in the FnOnce trait's arguments,
|
|
||||||
// type resolution always gives a "maybe" here.
|
|
||||||
if self.autoderef(span, ty).silence_errors().any(|(ty, _)| {
|
|
||||||
info!("check deref {:?} error", ty);
|
|
||||||
matches!(ty.kind(), ty::Error(_) | ty::Infer(_))
|
|
||||||
}) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.autoderef(span, ty).silence_errors().any(|(ty, _)| {
|
|
||||||
info!("check deref {:?} impl FnOnce", ty);
|
|
||||||
self.probe(|_| {
|
|
||||||
let trait_ref =
|
|
||||||
ty::TraitRef::new(tcx, fn_once, [ty, self.next_ty_var(span)]);
|
|
||||||
let poly_trait_ref = ty::Binder::dummy(trait_ref);
|
|
||||||
let obligation = Obligation::misc(
|
|
||||||
tcx,
|
|
||||||
span,
|
|
||||||
self.body_id,
|
|
||||||
self.param_env,
|
|
||||||
poly_trait_ref,
|
|
||||||
);
|
|
||||||
self.predicate_may_hold(&obligation)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_slice_ty(&self, ty: Ty<'tcx>, span: Span) -> bool {
|
fn is_slice_ty(&self, ty: Ty<'tcx>, span: Span) -> bool {
|
||||||
self.autoderef(span, ty)
|
self.autoderef(span, ty)
|
||||||
.silence_errors()
|
.silence_errors()
|
||||||
|
@ -2367,12 +2324,16 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
let is_accessible = field.vis.is_accessible_from(scope, tcx);
|
let is_accessible = field.vis.is_accessible_from(scope, tcx);
|
||||||
|
|
||||||
if is_accessible {
|
if is_accessible {
|
||||||
if self.is_fn_ty(field_ty, span) {
|
if let Some((what, _, _)) = self.extract_callable_info(field_ty) {
|
||||||
|
let what = match what {
|
||||||
|
DefIdOrName::DefId(def_id) => self.tcx.def_descr(def_id),
|
||||||
|
DefIdOrName::Name(what) => what,
|
||||||
|
};
|
||||||
let expr_span = expr.span.to(item_name.span);
|
let expr_span = expr.span.to(item_name.span);
|
||||||
err.multipart_suggestion(
|
err.multipart_suggestion(
|
||||||
format!(
|
format!(
|
||||||
"to call the function stored in `{item_name}`, \
|
"to call the {what} stored in `{item_name}`, \
|
||||||
surround the field access with parentheses",
|
surround the field access with parentheses",
|
||||||
),
|
),
|
||||||
vec![
|
vec![
|
||||||
(expr_span.shrink_to_lo(), '('.to_string()),
|
(expr_span.shrink_to_lo(), '('.to_string()),
|
||||||
|
|
|
@ -1075,93 +1075,110 @@ impl<'a, 'tcx> TypeErrCtxt<'a, 'tcx> {
|
||||||
) -> Option<(DefIdOrName, Ty<'tcx>, Vec<Ty<'tcx>>)> {
|
) -> Option<(DefIdOrName, Ty<'tcx>, Vec<Ty<'tcx>>)> {
|
||||||
// Autoderef is useful here because sometimes we box callables, etc.
|
// Autoderef is useful here because sometimes we box callables, etc.
|
||||||
let Some((def_id_or_name, output, inputs)) =
|
let Some((def_id_or_name, output, inputs)) =
|
||||||
(self.autoderef_steps)(found).into_iter().find_map(|(found, _)| {
|
(self.autoderef_steps)(found).into_iter().find_map(|(found, _)| match *found.kind() {
|
||||||
match *found.kind() {
|
ty::FnPtr(sig_tys, _) => Some((
|
||||||
ty::FnPtr(sig_tys, _) => Some((
|
DefIdOrName::Name("function pointer"),
|
||||||
DefIdOrName::Name("function pointer"),
|
sig_tys.output(),
|
||||||
sig_tys.output(),
|
sig_tys.inputs(),
|
||||||
sig_tys.inputs(),
|
)),
|
||||||
)),
|
ty::FnDef(def_id, _) => {
|
||||||
ty::FnDef(def_id, _) => {
|
let fn_sig = found.fn_sig(self.tcx);
|
||||||
let fn_sig = found.fn_sig(self.tcx);
|
Some((DefIdOrName::DefId(def_id), fn_sig.output(), fn_sig.inputs()))
|
||||||
Some((DefIdOrName::DefId(def_id), fn_sig.output(), fn_sig.inputs()))
|
}
|
||||||
}
|
ty::Closure(def_id, args) => {
|
||||||
ty::Closure(def_id, args) => {
|
let fn_sig = args.as_closure().sig();
|
||||||
let fn_sig = args.as_closure().sig();
|
Some((
|
||||||
Some((
|
DefIdOrName::DefId(def_id),
|
||||||
DefIdOrName::DefId(def_id),
|
fn_sig.output(),
|
||||||
fn_sig.output(),
|
fn_sig.inputs().map_bound(|inputs| inputs[0].tuple_fields().as_slice()),
|
||||||
fn_sig.inputs().map_bound(|inputs| &inputs[1..]),
|
))
|
||||||
))
|
}
|
||||||
}
|
ty::CoroutineClosure(def_id, args) => {
|
||||||
ty::Alias(ty::Opaque, ty::AliasTy { def_id, args, .. }) => {
|
let sig_parts = args.as_coroutine_closure().coroutine_closure_sig();
|
||||||
self.tcx
|
Some((
|
||||||
.item_super_predicates(def_id)
|
DefIdOrName::DefId(def_id),
|
||||||
.instantiate(self.tcx, args)
|
sig_parts.map_bound(|sig| {
|
||||||
.iter()
|
sig.to_coroutine(
|
||||||
.find_map(|pred| {
|
self.tcx,
|
||||||
if let ty::ClauseKind::Projection(proj) = pred.kind().skip_binder()
|
args.as_coroutine_closure().parent_args(),
|
||||||
&& self.tcx.is_lang_item(proj.projection_term.def_id,LangItem::FnOnceOutput)
|
// Just use infer vars here, since we don't really care
|
||||||
// args tuple will always be args[1]
|
// what these types are, just that we're returning a coroutine.
|
||||||
&& let ty::Tuple(args) = proj.projection_term.args.type_at(1).kind()
|
self.next_ty_var(DUMMY_SP),
|
||||||
{
|
self.tcx.coroutine_for_closure(def_id),
|
||||||
Some((
|
self.next_ty_var(DUMMY_SP),
|
||||||
DefIdOrName::DefId(def_id),
|
)
|
||||||
pred.kind().rebind(proj.term.expect_type()),
|
}),
|
||||||
pred.kind().rebind(args.as_slice()),
|
sig_parts.map_bound(|sig| sig.tupled_inputs_ty.tuple_fields().as_slice()),
|
||||||
))
|
))
|
||||||
} else {
|
}
|
||||||
None
|
ty::Alias(ty::Opaque, ty::AliasTy { def_id, args, .. }) => self
|
||||||
}
|
.tcx
|
||||||
})
|
.item_super_predicates(def_id)
|
||||||
}
|
.instantiate(self.tcx, args)
|
||||||
ty::Dynamic(data, _, ty::Dyn) => {
|
.iter()
|
||||||
data.iter().find_map(|pred| {
|
.find_map(|pred| {
|
||||||
if let ty::ExistentialPredicate::Projection(proj) = pred.skip_binder()
|
if let ty::ClauseKind::Projection(proj) = pred.kind().skip_binder()
|
||||||
|
&& self
|
||||||
|
.tcx
|
||||||
|
.is_lang_item(proj.projection_term.def_id, LangItem::FnOnceOutput)
|
||||||
|
// args tuple will always be args[1]
|
||||||
|
&& let ty::Tuple(args) = proj.projection_term.args.type_at(1).kind()
|
||||||
|
{
|
||||||
|
Some((
|
||||||
|
DefIdOrName::DefId(def_id),
|
||||||
|
pred.kind().rebind(proj.term.expect_type()),
|
||||||
|
pred.kind().rebind(args.as_slice()),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
ty::Dynamic(data, _, ty::Dyn) => data.iter().find_map(|pred| {
|
||||||
|
if let ty::ExistentialPredicate::Projection(proj) = pred.skip_binder()
|
||||||
&& self.tcx.is_lang_item(proj.def_id, LangItem::FnOnceOutput)
|
&& self.tcx.is_lang_item(proj.def_id, LangItem::FnOnceOutput)
|
||||||
// for existential projection, args are shifted over by 1
|
// for existential projection, args are shifted over by 1
|
||||||
&& let ty::Tuple(args) = proj.args.type_at(0).kind()
|
&& let ty::Tuple(args) = proj.args.type_at(0).kind()
|
||||||
{
|
{
|
||||||
Some((
|
Some((
|
||||||
DefIdOrName::Name("trait object"),
|
DefIdOrName::Name("trait object"),
|
||||||
pred.rebind(proj.term.expect_type()),
|
pred.rebind(proj.term.expect_type()),
|
||||||
pred.rebind(args.as_slice()),
|
pred.rebind(args.as_slice()),
|
||||||
))
|
))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
ty::Param(param) => {
|
}),
|
||||||
let generics = self.tcx.generics_of(body_id);
|
ty::Param(param) => {
|
||||||
let name = if generics.count() > param.index as usize
|
let generics = self.tcx.generics_of(body_id);
|
||||||
&& let def = generics.param_at(param.index as usize, self.tcx)
|
let name = if generics.count() > param.index as usize
|
||||||
&& matches!(def.kind, ty::GenericParamDefKind::Type { .. })
|
&& let def = generics.param_at(param.index as usize, self.tcx)
|
||||||
&& def.name == param.name
|
&& matches!(def.kind, ty::GenericParamDefKind::Type { .. })
|
||||||
|
&& def.name == param.name
|
||||||
|
{
|
||||||
|
DefIdOrName::DefId(def.def_id)
|
||||||
|
} else {
|
||||||
|
DefIdOrName::Name("type parameter")
|
||||||
|
};
|
||||||
|
param_env.caller_bounds().iter().find_map(|pred| {
|
||||||
|
if let ty::ClauseKind::Projection(proj) = pred.kind().skip_binder()
|
||||||
|
&& self
|
||||||
|
.tcx
|
||||||
|
.is_lang_item(proj.projection_term.def_id, LangItem::FnOnceOutput)
|
||||||
|
&& proj.projection_term.self_ty() == found
|
||||||
|
// args tuple will always be args[1]
|
||||||
|
&& let ty::Tuple(args) = proj.projection_term.args.type_at(1).kind()
|
||||||
{
|
{
|
||||||
DefIdOrName::DefId(def.def_id)
|
Some((
|
||||||
|
name,
|
||||||
|
pred.kind().rebind(proj.term.expect_type()),
|
||||||
|
pred.kind().rebind(args.as_slice()),
|
||||||
|
))
|
||||||
} else {
|
} else {
|
||||||
DefIdOrName::Name("type parameter")
|
None
|
||||||
};
|
}
|
||||||
param_env.caller_bounds().iter().find_map(|pred| {
|
})
|
||||||
if let ty::ClauseKind::Projection(proj) = pred.kind().skip_binder()
|
|
||||||
&& self.tcx.is_lang_item(proj.projection_term.def_id, LangItem::FnOnceOutput)
|
|
||||||
&& proj.projection_term.self_ty() == found
|
|
||||||
// args tuple will always be args[1]
|
|
||||||
&& let ty::Tuple(args) = proj.projection_term.args.type_at(1).kind()
|
|
||||||
{
|
|
||||||
Some((
|
|
||||||
name,
|
|
||||||
pred.kind().rebind(proj.term.expect_type()),
|
|
||||||
pred.kind().rebind(args.as_slice()),
|
|
||||||
))
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
}
|
||||||
|
_ => None,
|
||||||
})
|
})
|
||||||
else {
|
else {
|
||||||
return None;
|
return None;
|
||||||
|
|
7
tests/ui/closures/correct-args-on-call-suggestion.rs
Normal file
7
tests/ui/closures/correct-args-on-call-suggestion.rs
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
// Ensure we give the right args when we suggest calling a closure.
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let x = |a: i32, b: i32| a + b;
|
||||||
|
let y: i32 = x;
|
||||||
|
//~^ ERROR mismatched types
|
||||||
|
}
|
20
tests/ui/closures/correct-args-on-call-suggestion.stderr
Normal file
20
tests/ui/closures/correct-args-on-call-suggestion.stderr
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
error[E0308]: mismatched types
|
||||||
|
--> $DIR/correct-args-on-call-suggestion.rs:5:18
|
||||||
|
|
|
||||||
|
LL | let x = |a: i32, b: i32| a + b;
|
||||||
|
| ---------------- the found closure
|
||||||
|
LL | let y: i32 = x;
|
||||||
|
| --- ^ expected `i32`, found closure
|
||||||
|
| |
|
||||||
|
| expected due to this
|
||||||
|
|
|
||||||
|
= note: expected type `i32`
|
||||||
|
found closure `{closure@$DIR/correct-args-on-call-suggestion.rs:4:13: 4:29}`
|
||||||
|
help: use parentheses to call this closure
|
||||||
|
|
|
||||||
|
LL | let y: i32 = x(/* i32 */, /* i32 */);
|
||||||
|
| ++++++++++++++++++++++
|
||||||
|
|
||||||
|
error: aborting due to 1 previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0308`.
|
|
@ -7,7 +7,7 @@ LL | struct Obj<F> where F: FnMut() -> u32 {
|
||||||
LL | o.closure();
|
LL | o.closure();
|
||||||
| ^^^^^^^ field, not a method
|
| ^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `closure`, surround the field access with parentheses
|
help: to call the closure stored in `closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (o.closure)();
|
LL | (o.closure)();
|
||||||
| + +
|
| + +
|
||||||
|
|
|
@ -7,7 +7,7 @@ LL | struct Obj<F> where F: FnOnce() -> u32 {
|
||||||
LL | o_closure.closure();
|
LL | o_closure.closure();
|
||||||
| ^^^^^^^ field, not a method
|
| ^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `closure`, surround the field access with parentheses
|
help: to call the closure stored in `closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (o_closure.closure)();
|
LL | (o_closure.closure)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -46,7 +46,7 @@ LL | struct BoxedObj {
|
||||||
LL | boxed_fn.boxed_closure();
|
LL | boxed_fn.boxed_closure();
|
||||||
| ^^^^^^^^^^^^^ field, not a method
|
| ^^^^^^^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `boxed_closure`, surround the field access with parentheses
|
help: to call the trait object stored in `boxed_closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (boxed_fn.boxed_closure)();
|
LL | (boxed_fn.boxed_closure)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -60,7 +60,7 @@ LL | struct BoxedObj {
|
||||||
LL | boxed_closure.boxed_closure();
|
LL | boxed_closure.boxed_closure();
|
||||||
| ^^^^^^^^^^^^^ field, not a method
|
| ^^^^^^^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `boxed_closure`, surround the field access with parentheses
|
help: to call the trait object stored in `boxed_closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (boxed_closure.boxed_closure)();
|
LL | (boxed_closure.boxed_closure)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -99,7 +99,7 @@ LL | struct Obj<F> where F: FnOnce() -> u32 {
|
||||||
LL | check_expression().closure();
|
LL | check_expression().closure();
|
||||||
| ^^^^^^^ field, not a method
|
| ^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `closure`, surround the field access with parentheses
|
help: to call the trait object stored in `closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (check_expression().closure)();
|
LL | (check_expression().closure)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -113,7 +113,7 @@ LL | struct FuncContainer {
|
||||||
LL | (*self.container).f1(1);
|
LL | (*self.container).f1(1);
|
||||||
| ^^ field, not a method
|
| ^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `f1`, surround the field access with parentheses
|
help: to call the function pointer stored in `f1`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | ((*self.container).f1)(1);
|
LL | ((*self.container).f1)(1);
|
||||||
| + +
|
| + +
|
||||||
|
@ -127,7 +127,7 @@ LL | struct FuncContainer {
|
||||||
LL | (*self.container).f2(1);
|
LL | (*self.container).f2(1);
|
||||||
| ^^ field, not a method
|
| ^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `f2`, surround the field access with parentheses
|
help: to call the function pointer stored in `f2`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | ((*self.container).f2)(1);
|
LL | ((*self.container).f2)(1);
|
||||||
| + +
|
| + +
|
||||||
|
@ -141,7 +141,7 @@ LL | struct FuncContainer {
|
||||||
LL | (*self.container).f3(1);
|
LL | (*self.container).f3(1);
|
||||||
| ^^ field, not a method
|
| ^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `f3`, surround the field access with parentheses
|
help: to call the function pointer stored in `f3`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | ((*self.container).f3)(1);
|
LL | ((*self.container).f3)(1);
|
||||||
| + +
|
| + +
|
||||||
|
|
|
@ -7,7 +7,7 @@ LL | struct Example {
|
||||||
LL | demo.example(1);
|
LL | demo.example(1);
|
||||||
| ^^^^^^^ field, not a method
|
| ^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `example`, surround the field access with parentheses
|
help: to call the trait object stored in `example`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (demo.example)(1);
|
LL | (demo.example)(1);
|
||||||
| + +
|
| + +
|
||||||
|
|
|
@ -4,7 +4,7 @@ error[E0599]: no method named `closure` found for reference `&Obj<{closure@$DIR/
|
||||||
LL | p.closure();
|
LL | p.closure();
|
||||||
| ^^^^^^^ field, not a method
|
| ^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `closure`, surround the field access with parentheses
|
help: to call the closure stored in `closure`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (p.closure)();
|
LL | (p.closure)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -19,7 +19,7 @@ error[E0599]: no method named `fn_ptr` found for reference `&&Obj<{closure@$DIR/
|
||||||
LL | q.fn_ptr();
|
LL | q.fn_ptr();
|
||||||
| ^^^^^^ field, not a method
|
| ^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `fn_ptr`, surround the field access with parentheses
|
help: to call the function pointer stored in `fn_ptr`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (q.fn_ptr)();
|
LL | (q.fn_ptr)();
|
||||||
| + +
|
| + +
|
||||||
|
@ -30,7 +30,7 @@ error[E0599]: no method named `c_fn_ptr` found for reference `&D` in the current
|
||||||
LL | s.c_fn_ptr();
|
LL | s.c_fn_ptr();
|
||||||
| ^^^^^^^^ field, not a method
|
| ^^^^^^^^ field, not a method
|
||||||
|
|
|
|
||||||
help: to call the function stored in `c_fn_ptr`, surround the field access with parentheses
|
help: to call the function pointer stored in `c_fn_ptr`, surround the field access with parentheses
|
||||||
|
|
|
|
||||||
LL | (s.c_fn_ptr)();
|
LL | (s.c_fn_ptr)();
|
||||||
| + +
|
| + +
|
||||||
|
|
|
@ -31,6 +31,10 @@ note: required by a bound in `bar`
|
||||||
|
|
|
|
||||||
LL | fn bar(f: impl Future<Output=()>) {}
|
LL | fn bar(f: impl Future<Output=()>) {}
|
||||||
| ^^^^^^^^^^^^^^^^^ required by this bound in `bar`
|
| ^^^^^^^^^^^^^^^^^ required by this bound in `bar`
|
||||||
|
help: use parentheses to call this closure
|
||||||
|
|
|
||||||
|
LL | bar(async_closure());
|
||||||
|
| ++
|
||||||
|
|
||||||
error: aborting due to 2 previous errors
|
error: aborting due to 2 previous errors
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue