Skip to content

Commit e536257

Browse files
committed
Ensure tail expression will have a Ty for E0746
When the return type is `!Sized` we look for all the returned expressions in the body to fetch their types and provide a reasonable suggestion. The tail expression of the body is normally evaluated after checking whether the return type is `Sized`. Changing the order of the evaluation produces undesirable knock down effects, so we detect the specific case that newcomers are likely to encounter ,returning a single bare trait object, and only in that case we evaluate the tail expression's type so that the suggestion will be accurate.
1 parent d3c96f0 commit e536257

File tree

6 files changed

+113
-66
lines changed

6 files changed

+113
-66
lines changed

src/librustc_error_codes/error_codes/E0746.md

+1-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,7 @@ Return types cannot be `dyn Trait`s as they must be `Sized`.
22

33
Erroneous code example:
44

5-
```compile_fail,E0277
6-
# // FIXME: after E0746 is in beta, change the above
5+
```compile_fail,E0746
76
trait T {
87
fn bar(&self);
98
}

src/librustc_trait_selection/traits/error_reporting/suggestions.rs

+73-48
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ use rustc_hir::intravisit::Visitor;
1313
use rustc_hir::{AsyncGeneratorKind, GeneratorKind, Node};
1414
use rustc_middle::ty::TypeckTables;
1515
use rustc_middle::ty::{
16-
self, AdtKind, DefIdTree, ToPredicate, Ty, TyCtxt, TypeFoldable, WithConstness,
16+
self, AdtKind, DefIdTree, Infer, InferTy, ToPredicate, Ty, TyCtxt, TypeFoldable, WithConstness,
1717
};
1818
use rustc_span::symbol::{kw, sym, Symbol};
1919
use rustc_span::{MultiSpan, Span, DUMMY_SP};
@@ -826,12 +826,28 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
826826
.iter()
827827
.filter_map(|expr| tables.node_type_opt(expr.hir_id))
828828
.map(|ty| self.resolve_vars_if_possible(&ty));
829-
let (last_ty, all_returns_have_same_type, count) = ret_types.clone().fold(
830-
(None, true, 0),
831-
|(last_ty, mut same, count): (std::option::Option<Ty<'_>>, bool, usize), ty| {
829+
let (last_ty, all_returns_have_same_type, only_never_return) = ret_types.clone().fold(
830+
(None, true, true),
831+
|(last_ty, mut same, only_never_return): (std::option::Option<Ty<'_>>, bool, bool),
832+
ty| {
832833
let ty = self.resolve_vars_if_possible(&ty);
833-
same &= last_ty.map_or(true, |last_ty| last_ty == ty) && ty.kind != ty::Error;
834-
(Some(ty), same, count + 1)
834+
same &=
835+
ty.kind != ty::Error
836+
&& last_ty.map_or(true, |last_ty| {
837+
// FIXME: ideally we would use `can_coerce` here instead, but `typeck` comes
838+
// *after* in the dependency graph.
839+
match (&ty.kind, &last_ty.kind) {
840+
(Infer(InferTy::IntVar(_)), Infer(InferTy::IntVar(_)))
841+
| (Infer(InferTy::FloatVar(_)), Infer(InferTy::FloatVar(_)))
842+
| (Infer(InferTy::FreshIntTy(_)), Infer(InferTy::FreshIntTy(_)))
843+
| (
844+
Infer(InferTy::FreshFloatTy(_)),
845+
Infer(InferTy::FreshFloatTy(_)),
846+
) => true,
847+
_ => ty == last_ty,
848+
}
849+
});
850+
(Some(ty), same, only_never_return && matches!(ty.kind, ty::Never))
835851
},
836852
);
837853
let all_returns_conform_to_trait =
@@ -840,13 +856,14 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
840856
ty::Dynamic(predicates, _) => {
841857
let cause = ObligationCause::misc(ret_ty.span, ret_ty.hir_id);
842858
let param_env = ty::ParamEnv::empty();
843-
ret_types.all(|returned_ty| {
844-
predicates.iter().all(|predicate| {
845-
let pred = predicate.with_self_ty(self.tcx, returned_ty);
846-
let obl = Obligation::new(cause.clone(), param_env, pred);
847-
self.predicate_may_hold(&obl)
859+
only_never_return
860+
|| ret_types.all(|returned_ty| {
861+
predicates.iter().all(|predicate| {
862+
let pred = predicate.with_self_ty(self.tcx, returned_ty);
863+
let obl = Obligation::new(cause.clone(), param_env, pred);
864+
self.predicate_may_hold(&obl)
865+
})
848866
})
849-
}) || count == 0
850867
}
851868
_ => false,
852869
}
@@ -861,7 +878,7 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
861878
&ret_ty.kind,
862879
sm.span_to_snippet(ret_ty.span),
863880
// If any of the return types does not conform to the trait, then we can't
864-
// suggest `impl Trait` nor trait objects, it is a type mismatch error.
881+
// suggest `impl Trait` nor trait objects: it is a type mismatch error.
865882
all_returns_conform_to_trait,
866883
) {
867884
snippet
@@ -879,43 +896,15 @@ impl<'a, 'tcx> InferCtxtExt<'tcx> for InferCtxt<'a, 'tcx> {
879896
#using-trait-objects-that-allow-for-values-of-different-types>";
880897
let has_dyn = snippet.split_whitespace().next().map_or(false, |s| s == "dyn");
881898
let trait_obj = if has_dyn { &snippet[4..] } else { &snippet[..] };
882-
if count == 0 {
883-
// No return paths. Suggest `-> T`, `-> impl Trait`, and if `Trait` is object safe,
884-
// `-> Box<dyn Trait>`.
885-
err.note(
886-
"currently nothing is being returned, depending on the final implementation \
887-
you could change the return type in different ways",
888-
);
889-
err.span_suggestion(
890-
ret_ty.span,
891-
"you could use some type `T` that is `T: Sized` as the return type if all return \
892-
paths will have the same type",
893-
"T".to_string(),
894-
Applicability::MaybeIncorrect,
895-
);
896-
err.span_suggestion(
899+
if only_never_return {
900+
// No return paths, probably using `panic!()` or similar.
901+
// Suggest `-> T`, `-> impl Trait`, and if `Trait` is object safe, `-> Box<dyn Trait>`.
902+
suggest_trait_object_return_type_alternatives(
903+
err,
897904
ret_ty.span,
898-
&format!(
899-
"you could use `impl {}` as the return type if all return paths will have the \
900-
same type but you want to expose only the trait in the signature",
901-
trait_obj,
902-
),
903-
format!("impl {}", trait_obj),
904-
Applicability::MaybeIncorrect,
905+
trait_obj,
906+
is_object_safe,
905907
);
906-
err.note(impl_trait_msg);
907-
if is_object_safe {
908-
err.span_suggestion(
909-
ret_ty.span,
910-
&format!(
911-
"you could use a boxed trait object if all return paths `impl` trait `{}`",
912-
trait_obj,
913-
),
914-
format!("Box<dyn {}>", trait_obj),
915-
Applicability::MaybeIncorrect,
916-
);
917-
err.note(trait_obj_msg);
918-
}
919908
} else if let (Some(last_ty), true) = (last_ty, all_returns_have_same_type) {
920909
// Suggest `-> impl Trait`.
921910
err.span_suggestion(
@@ -1851,3 +1840,39 @@ impl NextTypeParamName for &[hir::GenericParam<'_>] {
18511840
.to_string()
18521841
}
18531842
}
1843+
1844+
fn suggest_trait_object_return_type_alternatives(
1845+
err: &mut DiagnosticBuilder<'tcx>,
1846+
ret_ty: Span,
1847+
trait_obj: &str,
1848+
is_object_safe: bool,
1849+
) {
1850+
err.span_suggestion(
1851+
ret_ty,
1852+
"use some type `T` that is `T: Sized` as the return type if all return paths have the \
1853+
same type",
1854+
"T".to_string(),
1855+
Applicability::MaybeIncorrect,
1856+
);
1857+
err.span_suggestion(
1858+
ret_ty,
1859+
&format!(
1860+
"use `impl {}` as the return type if all return paths have the same type but you \
1861+
want to expose only the trait in the signature",
1862+
trait_obj,
1863+
),
1864+
format!("impl {}", trait_obj),
1865+
Applicability::MaybeIncorrect,
1866+
);
1867+
if is_object_safe {
1868+
err.span_suggestion(
1869+
ret_ty,
1870+
&format!(
1871+
"use a boxed trait object if all return paths implement trait `{}`",
1872+
trait_obj,
1873+
),
1874+
format!("Box<dyn {}>", trait_obj),
1875+
Applicability::MaybeIncorrect,
1876+
);
1877+
}
1878+
}

src/librustc_typeck/check/mod.rs

+19-2
Original file line numberDiff line numberDiff line change
@@ -1305,7 +1305,6 @@ fn check_fn<'a, 'tcx>(
13051305
let hir = tcx.hir();
13061306

13071307
let declared_ret_ty = fn_sig.output();
1308-
fcx.require_type_is_sized(declared_ret_ty, decl.output.span(), traits::SizedReturnType);
13091308
let revealed_ret_ty =
13101309
fcx.instantiate_opaque_types_from_value(fn_id, &declared_ret_ty, decl.output.span());
13111310
debug!("check_fn: declared_ret_ty: {}, revealed_ret_ty: {}", declared_ret_ty, revealed_ret_ty);
@@ -1374,7 +1373,25 @@ fn check_fn<'a, 'tcx>(
13741373

13751374
inherited.tables.borrow_mut().liberated_fn_sigs_mut().insert(fn_id, fn_sig);
13761375

1377-
fcx.check_return_expr(&body.value);
1376+
if let ty::Dynamic(..) = declared_ret_ty.kind {
1377+
// FIXME: We need to verify that the return type is `Sized` after the return expression has
1378+
// been evaluated so that we have types available for all the nodes being returned, but that
1379+
// requires the coerced evaluated type to be stored. Moving `check_return_expr` before this
1380+
// causes unsized errors caused by the `declared_ret_ty` to point at the return expression,
1381+
// while keeping the current ordering we will ignore the tail expression's type because we
1382+
// don't know it yet. We can't do `check_expr_kind` while keeping `check_return_expr`
1383+
// because we will trigger "unreachable expression" lints unconditionally.
1384+
// Because of all of this, we perform a crude check to know whether the simplest `!Sized`
1385+
// case that a newcomer might make, returning a bare trait, and in that case we populate
1386+
// the tail expression's type so that the suggestion will be correct, but ignore all other
1387+
// possible cases.
1388+
fcx.check_expr(&body.value);
1389+
fcx.require_type_is_sized(declared_ret_ty, decl.output.span(), traits::SizedReturnType);
1390+
tcx.sess.delay_span_bug(decl.output.span(), "`!Sized` return type");
1391+
} else {
1392+
fcx.require_type_is_sized(declared_ret_ty, decl.output.span(), traits::SizedReturnType);
1393+
fcx.check_return_expr(&body.value);
1394+
}
13781395

13791396
// We insert the deferred_generator_interiors entry after visiting the body.
13801397
// This ensures that all nested generators appear before the entry of this generator.

src/test/ui/impl-trait/dyn-trait-return-should-be-impl-trait.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ fn bax() -> dyn Trait { //~ ERROR E0746
2626
if true {
2727
Struct
2828
} else {
29-
42
29+
42 //~ ERROR `if` and `else` have incompatible types
3030
}
3131
}
3232
fn bam() -> Box<dyn Trait> {

src/test/ui/impl-trait/dyn-trait-return-should-be-impl-trait.stderr

+16-7
Original file line numberDiff line numberDiff line change
@@ -72,18 +72,15 @@ error[E0746]: return type cannot have an unboxed trait object
7272
LL | fn bak() -> dyn Trait { unimplemented!() }
7373
| ^^^^^^^^^ doesn't have a size known at compile-time
7474
|
75-
= note: currently nothing is being returned, depending on the final implementation you could change the return type in different ways
76-
= note: for information on `impl Trait`, see <https://doc.rust-lang.org/book/ch10-02-traits.html#returning-types-that-implement-traits>
77-
= note: for information on trait objects, see <https://doc.rust-lang.org/book/ch17-02-trait-objects.html#using-trait-objects-that-allow-for-values-of-different-types>
78-
help: you could use some type `T` that is `T: Sized` as the return type if all return paths will have the same type
75+
help: use some type `T` that is `T: Sized` as the return type if all return paths have the same type
7976
|
8077
LL | fn bak() -> T { unimplemented!() }
8178
| ^
82-
help: you could use `impl Trait` as the return type if all return paths will have the same type but you want to expose only the trait in the signature
79+
help: use `impl Trait` as the return type if all return paths have the same type but you want to expose only the trait in the signature
8380
|
8481
LL | fn bak() -> impl Trait { unimplemented!() }
8582
| ^^^^^^^^^^
86-
help: you could use a boxed trait object if all return paths `impl` trait `Trait`
83+
help: use a boxed trait object if all return paths implement trait `Trait`
8784
|
8885
LL | fn bak() -> Box<dyn Trait> { unimplemented!() }
8986
| ^^^^^^^^^^^^^^
@@ -107,6 +104,18 @@ LL | }
107104
LL | Box::new(42)
108105
|
109106

107+
error[E0308]: `if` and `else` have incompatible types
108+
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:29:9
109+
|
110+
LL | / if true {
111+
LL | | Struct
112+
| | ------ expected because of this
113+
LL | | } else {
114+
LL | | 42
115+
| | ^^ expected struct `Struct`, found integer
116+
LL | | }
117+
| |_____- `if` and `else` have incompatible types
118+
110119
error[E0746]: return type cannot have an unboxed trait object
111120
--> $DIR/dyn-trait-return-should-be-impl-trait.rs:25:13
112121
|
@@ -278,7 +287,7 @@ help: use `impl Trait` as the return type, as all return paths are of type `{int
278287
LL | fn bay() -> impl Trait {
279288
| ^^^^^^^^^^
280289

281-
error: aborting due to 19 previous errors
290+
error: aborting due to 20 previous errors
282291

283292
Some errors have detailed explanations: E0277, E0308, E0746.
284293
For more information about an error, try `rustc --explain E0277`.

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

+3-6
Original file line numberDiff line numberDiff line change
@@ -4,18 +4,15 @@ error[E0746]: return type cannot have an unboxed trait object
44
LL | dyn AbstractRenderer
55
| ^^^^^^^^^^^^^^^^^^^^ doesn't have a size known at compile-time
66
|
7-
= note: currently nothing is being returned, depending on the final implementation you could change the return type in different ways
8-
= note: for information on `impl Trait`, see <https://doc.rust-lang.org/book/ch10-02-traits.html#returning-types-that-implement-traits>
9-
= note: for information on trait objects, see <https://doc.rust-lang.org/book/ch17-02-trait-objects.html#using-trait-objects-that-allow-for-values-of-different-types>
10-
help: you could use some type `T` that is `T: Sized` as the return type if all return paths will have the same type
7+
help: use some type `T` that is `T: Sized` as the return type if all return paths have the same type
118
|
129
LL | T
1310
|
14-
help: you could use `impl AbstractRenderer` as the return type if all return paths will have the same type but you want to expose only the trait in the signature
11+
help: use `impl AbstractRenderer` as the return type if all return paths have the same type but you want to expose only the trait in the signature
1512
|
1613
LL | impl AbstractRenderer
1714
|
18-
help: you could use a boxed trait object if all return paths `impl` trait `AbstractRenderer`
15+
help: use a boxed trait object if all return paths implement trait `AbstractRenderer`
1916
|
2017
LL | Box<dyn AbstractRenderer>
2118
|

0 commit comments

Comments
 (0)