Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit cdf5394

Browse files
committedNov 18, 2018
Forbid impl Trait from referring to unnamable recursive types
There is no type T, such that `T = [T; 2]`, we should not allow this to be circumvented by impl Trait.
1 parent 0d0d732 commit cdf5394

File tree

6 files changed

+316
-11
lines changed

6 files changed

+316
-11
lines changed
 

‎src/librustc/ty/util.rs

+68-2
Original file line numberDiff line numberDiff line change
@@ -17,15 +17,15 @@ use hir::{self, Node};
1717
use ich::NodeIdHashingMode;
1818
use traits::{self, ObligationCause};
1919
use ty::{self, Ty, TyCtxt, GenericParamDefKind, TypeFoldable};
20-
use ty::subst::{Substs, UnpackedKind};
20+
use ty::subst::{Subst, Substs, UnpackedKind};
2121
use ty::query::TyCtxtAt;
2222
use ty::TyKind::*;
2323
use ty::layout::{Integer, IntegerExt};
2424
use util::common::ErrorReported;
2525
use middle::lang_items;
2626

2727
use rustc_data_structures::stable_hasher::{StableHasher, HashStable};
28-
use rustc_data_structures::fx::FxHashMap;
28+
use rustc_data_structures::fx::{FxHashMap, FxHashSet};
2929
use std::{cmp, fmt};
3030
use syntax::ast;
3131
use syntax::attr::{self, SignedInt, UnsignedInt};
@@ -628,6 +628,72 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
628628
}
629629
}
630630
}
631+
632+
/// Expands the impl trait, stopping if the type is recursive.
633+
pub fn try_expand_impl_trait_type(
634+
self,
635+
def_id: DefId,
636+
substs: &'tcx Substs<'tcx>,
637+
) -> Result<Ty<'tcx>, Ty<'tcx>> {
638+
use crate::ty::fold::TypeFolder;
639+
640+
struct OpaqueTypeExpander<'a, 'gcx, 'tcx> {
641+
seen_opaque_tys: FxHashSet<DefId>,
642+
primary_def_id: DefId,
643+
found_recursion: bool,
644+
tcx: TyCtxt<'a, 'gcx, 'tcx>,
645+
}
646+
647+
impl<'a, 'gcx, 'tcx> OpaqueTypeExpander<'a, 'gcx, 'tcx> {
648+
fn expand_opaque_ty(
649+
&mut self,
650+
def_id: DefId,
651+
substs: &'tcx Substs<'tcx>,
652+
) -> Option<Ty<'tcx>> {
653+
if self.found_recursion {
654+
None
655+
} else if self.seen_opaque_tys.insert(def_id) {
656+
let generic_ty = self.tcx.type_of(def_id);
657+
let concrete_ty = generic_ty.subst(self.tcx, substs);
658+
let expanded_ty = self.fold_ty(concrete_ty);
659+
self.seen_opaque_tys.remove(&def_id);
660+
Some(expanded_ty)
661+
} else {
662+
// If another opaque type that we contain is recursive, then it
663+
// will report the error, so we don't have to.
664+
self.found_recursion = def_id == self.primary_def_id;
665+
None
666+
}
667+
}
668+
}
669+
670+
impl<'a, 'gcx, 'tcx> TypeFolder<'gcx, 'tcx> for OpaqueTypeExpander<'a, 'gcx, 'tcx> {
671+
fn tcx(&self) -> TyCtxt<'_, 'gcx, 'tcx> {
672+
self.tcx
673+
}
674+
675+
fn fold_ty(&mut self, t: Ty<'tcx>) -> Ty<'tcx> {
676+
if let ty::Opaque(def_id, substs) = t.sty {
677+
self.expand_opaque_ty(def_id, substs).unwrap_or(t)
678+
} else {
679+
t.super_fold_with(self)
680+
}
681+
}
682+
}
683+
684+
let mut visitor = OpaqueTypeExpander {
685+
seen_opaque_tys: FxHashSet::default(),
686+
primary_def_id: def_id,
687+
found_recursion: false,
688+
tcx: self,
689+
};
690+
let expanded_type = visitor.expand_opaque_ty(def_id, substs).unwrap();
691+
if visitor.found_recursion {
692+
Err(expanded_type)
693+
} else {
694+
Ok(expanded_type)
695+
}
696+
}
631697
}
632698

633699
impl<'a, 'tcx> ty::TyS<'tcx> {

‎src/librustc_typeck/check/mod.rs

+28-1
Original file line numberDiff line numberDiff line change
@@ -1306,6 +1306,24 @@ fn check_union<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
13061306
check_packed(tcx, span, def_id);
13071307
}
13081308

1309+
fn check_opaque<'a, 'tcx>(
1310+
tcx: TyCtxt<'a, 'tcx, 'tcx>,
1311+
def_id: DefId,
1312+
substs: &'tcx Substs<'tcx>,
1313+
span: Span,
1314+
) {
1315+
if let Err(partially_expanded_type) = tcx.try_expand_impl_trait_type(def_id, substs) {
1316+
let mut err = tcx.sess.struct_span_err(span, "recursive opaque type");
1317+
err.span_label(span, "resolves to self-referential type");
1318+
if let ty::Opaque(..) = partially_expanded_type.sty {
1319+
err.note("type resolves to itself");
1320+
} else {
1321+
err.note(&format!("resolved type is `{}`", partially_expanded_type));
1322+
}
1323+
err.emit();
1324+
}
1325+
}
1326+
13091327
pub fn check_item_type<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>, it: &'tcx hir::Item) {
13101328
debug!(
13111329
"check_item_type(it.id={}, it.name={})",
@@ -1352,7 +1370,16 @@ pub fn check_item_type<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>, it: &'tcx hir::Ite
13521370
hir::ItemKind::Union(..) => {
13531371
check_union(tcx, it.id, it.span);
13541372
}
1355-
hir::ItemKind::Existential(..) | hir::ItemKind::Ty(..) => {
1373+
hir::ItemKind::Existential(..) => {
1374+
let def_id = tcx.hir.local_def_id(it.id);
1375+
let pty_ty = tcx.type_of(def_id);
1376+
let generics = tcx.generics_of(def_id);
1377+
1378+
check_bounds_are_used(tcx, &generics, pty_ty);
1379+
let substs = Substs::identity_for_item(tcx, def_id);
1380+
check_opaque(tcx, def_id, substs, it.span);
1381+
}
1382+
hir::ItemKind::Ty(..) => {
13561383
let def_id = tcx.hir.local_def_id(it.id);
13571384
let pty_ty = tcx.type_of(def_id);
13581385
let generics = tcx.generics_of(def_id);

‎src/test/ui/impl-trait/infinite-impl-trait-issue-38064.rs

+2-4
Original file line numberDiff line numberDiff line change
@@ -13,17 +13,15 @@
1313
//
1414
// Regression test for #38064.
1515

16-
// error-pattern:overflow evaluating the requirement `impl Quux`
17-
1816
trait Quux {}
1917

20-
fn foo() -> impl Quux {
18+
fn foo() -> impl Quux { //~ recursive opaque type
2119
struct Foo<T>(T);
2220
impl<T> Quux for Foo<T> {}
2321
Foo(bar())
2422
}
2523

26-
fn bar() -> impl Quux {
24+
fn bar() -> impl Quux { //~ recursive opaque type
2725
struct Bar<T>(T);
2826
impl<T> Quux for Bar<T> {}
2927
Bar(foo())

‎src/test/ui/impl-trait/infinite-impl-trait-issue-38064.stderr

+15-4
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,18 @@
1-
error[E0275]: overflow evaluating the requirement `impl Quux`
1+
error: recursive opaque type
2+
--> $DIR/infinite-impl-trait-issue-38064.rs:18:13
23
|
3-
= help: consider adding a `#![recursion_limit="128"]` attribute to your crate
4+
LL | fn foo() -> impl Quux { //~ recursive opaque type
5+
| ^^^^^^^^^ resolves to self-referential type
6+
|
7+
= note: resolved type is `foo::Foo<bar::Bar<impl Quux>>`
8+
9+
error: recursive opaque type
10+
--> $DIR/infinite-impl-trait-issue-38064.rs:24:13
11+
|
12+
LL | fn bar() -> impl Quux { //~ recursive opaque type
13+
| ^^^^^^^^^ resolves to self-referential type
14+
|
15+
= note: resolved type is `bar::Bar<foo::Foo<impl Quux>>`
416

5-
error: aborting due to previous error
17+
error: aborting due to 2 previous errors
618

7-
For more information about this error, try `rustc --explain E0275`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
// Test that impl trait does not allow creating recursive types that are
2+
// otherwise forbidden.
3+
4+
#![feature(await_macro, async_await, futures_api, generators)]
5+
6+
fn option(i: i32) -> impl Sized { //~ ERROR
7+
if i < 0 {
8+
None
9+
} else {
10+
Some((option(i - 1), i))
11+
}
12+
}
13+
14+
fn tuple() -> impl Sized { //~ ERROR
15+
(tuple(),)
16+
}
17+
18+
fn array() -> impl Sized { //~ ERROR
19+
[array()]
20+
}
21+
22+
fn ptr() -> impl Sized { //~ ERROR
23+
&ptr() as *const _
24+
}
25+
26+
fn fn_ptr() -> impl Sized { //~ ERROR
27+
fn_ptr as fn() -> _
28+
}
29+
30+
fn closure_capture() -> impl Sized { //~ ERROR
31+
let x = closure_capture();
32+
move || { x; }
33+
}
34+
35+
fn closure_ref_capture() -> impl Sized { //~ ERROR
36+
let x = closure_ref_capture();
37+
move || { &x; }
38+
}
39+
40+
fn closure_sig() -> impl Sized { //~ ERROR
41+
|| closure_sig()
42+
}
43+
44+
fn generator_sig() -> impl Sized { //~ ERROR
45+
|| generator_sig()
46+
}
47+
48+
fn generator_capture() -> impl Sized { //~ ERROR
49+
let x = generator_capture();
50+
move || { yield; x; }
51+
}
52+
53+
fn substs_change<T>() -> impl Sized { //~ ERROR
54+
(substs_change::<&T>(),)
55+
}
56+
57+
fn generator_hold() -> impl Sized { //~ ERROR
58+
move || {
59+
let x = generator_hold();
60+
yield;
61+
x;
62+
}
63+
}
64+
65+
async fn recursive_async_function() -> () { //~ ERROR
66+
await!(recursive_async_function());
67+
}
68+
69+
fn use_fn_ptr() -> impl Sized { // OK, error already reported
70+
fn_ptr()
71+
}
72+
73+
fn mutual_recursion() -> impl Sync { //~ ERROR
74+
mutual_recursion_b()
75+
}
76+
77+
fn mutual_recursion_b() -> impl Sized { //~ ERROR
78+
mutual_recursion()
79+
}
80+
81+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,122 @@
1+
error: recursive opaque type
2+
--> $DIR/recursive-impl-trait-type.rs:6:22
3+
|
4+
LL | fn option(i: i32) -> impl Sized { //~ ERROR
5+
| ^^^^^^^^^^ resolves to self-referential type
6+
|
7+
= note: resolved type is `std::option::Option<(impl Sized, i32)>`
8+
9+
error: recursive opaque type
10+
--> $DIR/recursive-impl-trait-type.rs:14:15
11+
|
12+
LL | fn tuple() -> impl Sized { //~ ERROR
13+
| ^^^^^^^^^^ resolves to self-referential type
14+
|
15+
= note: resolved type is `(impl Sized,)`
16+
17+
error: recursive opaque type
18+
--> $DIR/recursive-impl-trait-type.rs:18:15
19+
|
20+
LL | fn array() -> impl Sized { //~ ERROR
21+
| ^^^^^^^^^^ resolves to self-referential type
22+
|
23+
= note: resolved type is `[impl Sized; 1]`
24+
25+
error: recursive opaque type
26+
--> $DIR/recursive-impl-trait-type.rs:22:13
27+
|
28+
LL | fn ptr() -> impl Sized { //~ ERROR
29+
| ^^^^^^^^^^ resolves to self-referential type
30+
|
31+
= note: resolved type is `*const impl Sized`
32+
33+
error: recursive opaque type
34+
--> $DIR/recursive-impl-trait-type.rs:26:16
35+
|
36+
LL | fn fn_ptr() -> impl Sized { //~ ERROR
37+
| ^^^^^^^^^^ resolves to self-referential type
38+
|
39+
= note: resolved type is `fn() -> impl Sized`
40+
41+
error: recursive opaque type
42+
--> $DIR/recursive-impl-trait-type.rs:30:25
43+
|
44+
LL | fn closure_capture() -> impl Sized { //~ ERROR
45+
| ^^^^^^^^^^ resolves to self-referential type
46+
|
47+
= note: resolved type is `[closure@$DIR/recursive-impl-trait-type.rs:32:5: 32:19 x:impl Sized]`
48+
49+
error: recursive opaque type
50+
--> $DIR/recursive-impl-trait-type.rs:35:29
51+
|
52+
LL | fn closure_ref_capture() -> impl Sized { //~ ERROR
53+
| ^^^^^^^^^^ resolves to self-referential type
54+
|
55+
= note: resolved type is `[closure@$DIR/recursive-impl-trait-type.rs:37:5: 37:20 x:impl Sized]`
56+
57+
error: recursive opaque type
58+
--> $DIR/recursive-impl-trait-type.rs:40:21
59+
|
60+
LL | fn closure_sig() -> impl Sized { //~ ERROR
61+
| ^^^^^^^^^^ resolves to self-referential type
62+
|
63+
= note: resolved type is `[closure@$DIR/recursive-impl-trait-type.rs:41:5: 41:21]`
64+
65+
error: recursive opaque type
66+
--> $DIR/recursive-impl-trait-type.rs:44:23
67+
|
68+
LL | fn generator_sig() -> impl Sized { //~ ERROR
69+
| ^^^^^^^^^^ resolves to self-referential type
70+
|
71+
= note: resolved type is `[closure@$DIR/recursive-impl-trait-type.rs:45:5: 45:23]`
72+
73+
error: recursive opaque type
74+
--> $DIR/recursive-impl-trait-type.rs:48:27
75+
|
76+
LL | fn generator_capture() -> impl Sized { //~ ERROR
77+
| ^^^^^^^^^^ resolves to self-referential type
78+
|
79+
= note: resolved type is `[generator@$DIR/recursive-impl-trait-type.rs:50:5: 50:26 x:impl Sized {()}]`
80+
81+
error: recursive opaque type
82+
--> $DIR/recursive-impl-trait-type.rs:53:26
83+
|
84+
LL | fn substs_change<T>() -> impl Sized { //~ ERROR
85+
| ^^^^^^^^^^ resolves to self-referential type
86+
|
87+
= note: resolved type is `(impl Sized,)`
88+
89+
error: recursive opaque type
90+
--> $DIR/recursive-impl-trait-type.rs:57:24
91+
|
92+
LL | fn generator_hold() -> impl Sized { //~ ERROR
93+
| ^^^^^^^^^^ resolves to self-referential type
94+
|
95+
= note: resolved type is `[generator@$DIR/recursive-impl-trait-type.rs:58:5: 62:6 {impl Sized, ()}]`
96+
97+
error: recursive opaque type
98+
--> $DIR/recursive-impl-trait-type.rs:65:40
99+
|
100+
LL | async fn recursive_async_function() -> () { //~ ERROR
101+
| ^^ resolves to self-referential type
102+
|
103+
= note: resolved type is `std::future::GenFuture<[static generator@$DIR/recursive-impl-trait-type.rs:65:43: 67:2 {impl std::future::Future, ()}]>`
104+
105+
error: recursive opaque type
106+
--> $DIR/recursive-impl-trait-type.rs:73:26
107+
|
108+
LL | fn mutual_recursion() -> impl Sync { //~ ERROR
109+
| ^^^^^^^^^ resolves to self-referential type
110+
|
111+
= note: type resolves to itself
112+
113+
error: recursive opaque type
114+
--> $DIR/recursive-impl-trait-type.rs:77:28
115+
|
116+
LL | fn mutual_recursion_b() -> impl Sized { //~ ERROR
117+
| ^^^^^^^^^^ resolves to self-referential type
118+
|
119+
= note: type resolves to itself
120+
121+
error: aborting due to 15 previous errors
122+

0 commit comments

Comments
 (0)
Please sign in to comment.