Skip to content

Commit 098cf88

Browse files
committed
Auto merge of rust-lang#101359 - compiler-errors:cannot-call-trait-object-with-unsized-return, r=lcnr
Point out when a callable is not actually callable because its return is not sized Fixes rust-lang#100755 I didn't add a UI test for that one because it's equivalent to the UI test that already exists in the suite.
2 parents 6c358c6 + 1254b32 commit 098cf88

File tree

5 files changed

+47
-19
lines changed

5 files changed

+47
-19
lines changed

compiler/rustc_trait_selection/src/infer.rs

+17
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,13 @@ pub trait InferCtxtExt<'tcx> {
2424
span: Span,
2525
) -> bool;
2626

27+
fn type_is_sized_modulo_regions(
28+
&self,
29+
param_env: ty::ParamEnv<'tcx>,
30+
ty: Ty<'tcx>,
31+
span: Span,
32+
) -> bool;
33+
2734
fn partially_normalize_associated_types_in<T>(
2835
&self,
2936
cause: ObligationCause<'tcx>,
@@ -74,6 +81,16 @@ impl<'cx, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'cx, 'tcx> {
7481
traits::type_known_to_meet_bound_modulo_regions(self, param_env, ty, copy_def_id, span)
7582
}
7683

84+
fn type_is_sized_modulo_regions(
85+
&self,
86+
param_env: ty::ParamEnv<'tcx>,
87+
ty: Ty<'tcx>,
88+
span: Span,
89+
) -> bool {
90+
let lang_item = self.tcx.require_lang_item(LangItem::Sized, None);
91+
traits::type_known_to_meet_bound_modulo_regions(self, param_env, ty, lang_item, span)
92+
}
93+
7794
/// Normalizes associated types in `value`, potentially returning
7895
/// new obligations that must further be processed.
7996
fn partially_normalize_associated_types_in<T>(

compiler/rustc_typeck/src/check/callee.rs

+22-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use super::method::MethodCallee;
2-
use super::{Expectation, FnCtxt, TupleArgumentsFlag};
2+
use super::{DefIdOrName, Expectation, FnCtxt, TupleArgumentsFlag};
33
use crate::type_error_struct;
44

55
use rustc_errors::{struct_span_err, Applicability, Diagnostic};
@@ -24,7 +24,8 @@ use rustc_span::symbol::{sym, Ident};
2424
use rustc_span::Span;
2525
use rustc_target::spec::abi;
2626
use rustc_trait_selection::autoderef::Autoderef;
27-
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt;
27+
use rustc_trait_selection::infer::InferCtxtExt as _;
28+
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _;
2829

2930
use std::iter;
3031

@@ -471,7 +472,25 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
471472
};
472473

473474
if !self.maybe_suggest_bad_array_definition(&mut err, call_expr, callee_expr) {
474-
err.span_label(call_expr.span, "call expression requires function");
475+
if let Some((maybe_def, output_ty, _)) = self.extract_callable_info(callee_expr, callee_ty)
476+
&& !self.type_is_sized_modulo_regions(self.param_env, output_ty, callee_expr.span)
477+
{
478+
let descr = match maybe_def {
479+
DefIdOrName::DefId(def_id) => self.tcx.def_kind(def_id).descr(def_id),
480+
DefIdOrName::Name(name) => name,
481+
};
482+
err.span_label(
483+
callee_expr.span,
484+
format!("this {descr} returns an unsized value `{output_ty}`, so it cannot be called")
485+
);
486+
if let DefIdOrName::DefId(def_id) = maybe_def
487+
&& let Some(def_span) = self.tcx.hir().span_if_local(def_id)
488+
{
489+
err.span_label(def_span, "the callable type is defined here");
490+
}
491+
} else {
492+
err.span_label(call_expr.span, "call expression requires function");
493+
}
475494
}
476495

477496
if let Some(span) = self.tcx.hir().res_span(def) {

compiler/rustc_typeck/src/check/cast.rs

+2-11
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,6 @@ use crate::type_error_struct;
3535
use hir::def_id::LOCAL_CRATE;
3636
use rustc_errors::{struct_span_err, Applicability, DiagnosticBuilder, ErrorGuaranteed};
3737
use rustc_hir as hir;
38-
use rustc_hir::lang_items::LangItem;
3938
use rustc_middle::mir::Mutability;
4039
use rustc_middle::ty::adjustment::AllowTwoPhase;
4140
use rustc_middle::ty::cast::{CastKind, CastTy};
@@ -47,7 +46,6 @@ use rustc_session::Session;
4746
use rustc_span::symbol::sym;
4847
use rustc_span::Span;
4948
use rustc_trait_selection::infer::InferCtxtExt;
50-
use rustc_trait_selection::traits;
5149
use rustc_trait_selection::traits::error_reporting::report_object_safety_error;
5250

5351
/// Reifies a cast check to be checked once we have full type information for
@@ -97,7 +95,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
9795
return Err(reported);
9896
}
9997

100-
if self.type_is_known_to_be_sized_modulo_regions(t, span) {
98+
if self.type_is_sized_modulo_regions(self.param_env, t, span) {
10199
return Ok(Some(PointerKind::Thin));
102100
}
103101

@@ -705,7 +703,7 @@ impl<'a, 'tcx> CastCheck<'tcx> {
705703

706704
debug!("check_cast({}, {:?} as {:?})", self.expr.hir_id, self.expr_ty, self.cast_ty);
707705

708-
if !fcx.type_is_known_to_be_sized_modulo_regions(self.cast_ty, self.span)
706+
if !fcx.type_is_sized_modulo_regions(fcx.param_env, self.cast_ty, self.span)
709707
&& !self.cast_ty.has_infer_types()
710708
{
711709
self.report_cast_to_unsized_type(fcx);
@@ -1084,10 +1082,3 @@ impl<'a, 'tcx> CastCheck<'tcx> {
10841082
);
10851083
}
10861084
}
1087-
1088-
impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
1089-
fn type_is_known_to_be_sized_modulo_regions(&self, ty: Ty<'tcx>, span: Span) -> bool {
1090-
let lang_item = self.tcx.require_lang_item(LangItem::Sized, None);
1091-
traits::type_known_to_meet_bound_modulo_regions(self, self.param_env, ty, lang_item, span)
1092-
}
1093-
}

compiler/rustc_typeck/src/check/fn_ctxt/suggestions.rs

+5-2
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,10 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
144144
false
145145
}
146146

147-
fn extract_callable_info(
147+
/// Extracts information about a callable type for diagnostics. This is a
148+
/// heuristic -- it doesn't necessarily mean that a type is always callable,
149+
/// because the callable type must also be well-formed to be called.
150+
pub(in super::super) fn extract_callable_info(
148151
&self,
149152
expr: &Expr<'_>,
150153
found: Ty<'tcx>,
@@ -1130,7 +1133,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
11301133
}
11311134
}
11321135

1133-
enum DefIdOrName {
1136+
pub enum DefIdOrName {
11341137
DefId(DefId),
11351138
Name(&'static str),
11361139
}

src/test/ui/issues/issue-41139.stderr

+1-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,7 @@ LL | fn get_function<'a>() -> &'a dyn Fn() -> dyn Trait {
55
| -------------------------------------------------- `get_function` defined here returns `&dyn Fn() -> (dyn Trait + 'static)`
66
...
77
LL | let t: &dyn Trait = &get_function()();
8-
| ^^^^^^^^^^^^^^--
9-
| |
10-
| call expression requires function
8+
| ^^^^^^^^^^^^^^ this trait object returns an unsized value `(dyn Trait + 'static)`, so it cannot be called
119

1210
error: aborting due to previous error
1311

0 commit comments

Comments
 (0)