From 2bb9c5875d0ab956f7d87c60de34bc3f2a427c1e Mon Sep 17 00:00:00 2001 From: Andrew Cann Date: Thu, 9 Feb 2017 17:34:45 +0800 Subject: [PATCH 1/2] Add recursion limit to inhabitedness check Fixes #39489. Add test aswell. --- src/librustc/ty/inhabitedness/mod.rs | 36 +++++++++++++------ src/librustc/ty/sty.rs | 2 +- src/librustc_const_eval/_match.rs | 2 +- src/librustc_mir/build/matches/simplify.rs | 2 +- .../inhabitedness-infinite-loop.rs | 24 +++++++++++++ 5 files changed, 52 insertions(+), 14 deletions(-) create mode 100644 src/test/compile-fail/inhabitedness-infinite-loop.rs diff --git a/src/librustc/ty/inhabitedness/mod.rs b/src/librustc/ty/inhabitedness/mod.rs index 18a3f1a218d85..51d046728c2bf 100644 --- a/src/librustc/ty/inhabitedness/mod.rs +++ b/src/librustc/ty/inhabitedness/mod.rs @@ -62,11 +62,14 @@ mod def_id_forest; // This code should only compile in modules where the uninhabitedness of Foo is // visible. +const ARBITRARY_RECURSION_LIMIT: u32 = 24; + impl<'a, 'gcx, 'tcx> AdtDef { /// Calculate the forest of DefIds from which this adt is visibly uninhabited. pub fn uninhabited_from( &self, visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, + recursion_depth: u32, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>) -> DefIdForest { @@ -75,7 +78,7 @@ impl<'a, 'gcx, 'tcx> AdtDef { } let ret = DefIdForest::intersection(tcx, self.variants.iter().map(|v| { - v.uninhabited_from(visited, tcx, substs, self.adt_kind()) + v.uninhabited_from(visited, recursion_depth, tcx, substs, self.adt_kind()) })); visited.remove(&(self.did, substs)); ret @@ -87,6 +90,7 @@ impl<'a, 'gcx, 'tcx> VariantDef { pub fn uninhabited_from( &self, visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, + recursion_depth: u32, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>, adt_kind: AdtKind) -> DefIdForest @@ -94,17 +98,17 @@ impl<'a, 'gcx, 'tcx> VariantDef { match adt_kind { AdtKind::Union => { DefIdForest::intersection(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, tcx, substs, false) + f.uninhabited_from(visited, recursion_depth, tcx, substs, false) })) }, AdtKind::Struct => { DefIdForest::union(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, tcx, substs, false) + f.uninhabited_from(visited, recursion_depth, tcx, substs, false) })) }, AdtKind::Enum => { DefIdForest::union(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, tcx, substs, true) + f.uninhabited_from(visited, recursion_depth, tcx, substs, true) })) }, } @@ -116,11 +120,14 @@ impl<'a, 'gcx, 'tcx> FieldDef { pub fn uninhabited_from( &self, visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, + recursion_depth: u32, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>, is_enum: bool) -> DefIdForest { - let mut data_uninhabitedness = move || self.ty(tcx, substs).uninhabited_from(visited, tcx); + let mut data_uninhabitedness = move || { + self.ty(tcx, substs).uninhabited_from(visited, recursion_depth, tcx) + }; // FIXME(canndrew): Currently enum fields are (incorrectly) stored with // Visibility::Invisible so we need to override self.vis if we're // dealing with an enum. @@ -145,8 +152,14 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { pub fn uninhabited_from( &self, visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, + mut recursion_depth: u32, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> DefIdForest { + recursion_depth += 1; + if recursion_depth >= ARBITRARY_RECURSION_LIMIT { + return DefIdForest::empty(); + } + match tcx.lift_to_global(&self) { Some(global_ty) => { { @@ -155,13 +168,13 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { return forest.clone(); } } - let forest = global_ty.uninhabited_from_inner(visited, tcx); + let forest = global_ty.uninhabited_from_inner(visited, recursion_depth, tcx); let mut cache = tcx.inhabitedness_cache.borrow_mut(); cache.insert(global_ty, forest.clone()); forest }, None => { - let forest = self.uninhabited_from_inner(visited, tcx); + let forest = self.uninhabited_from_inner(visited, recursion_depth, tcx); forest }, } @@ -170,28 +183,29 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { fn uninhabited_from_inner( &self, visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, + recursion_depth: u32, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> DefIdForest { match self.sty { TyAdt(def, substs) => { - def.uninhabited_from(visited, tcx, substs) + def.uninhabited_from(visited, recursion_depth, tcx, substs) }, TyNever => DefIdForest::full(tcx), TyTuple(ref tys, _) => { DefIdForest::union(tcx, tys.iter().map(|ty| { - ty.uninhabited_from(visited, tcx) + ty.uninhabited_from(visited, recursion_depth, tcx) })) }, TyArray(ty, len) => { if len == 0 { DefIdForest::empty() } else { - ty.uninhabited_from(visited, tcx) + ty.uninhabited_from(visited, recursion_depth, tcx) } } TyRef(_, ref tm) => { - tm.ty.uninhabited_from(visited, tcx) + tm.ty.uninhabited_from(visited, recursion_depth, tcx) } _ => DefIdForest::empty(), diff --git a/src/librustc/ty/sty.rs b/src/librustc/ty/sty.rs index 4ce1d7a901362..61b774924260d 100644 --- a/src/librustc/ty/sty.rs +++ b/src/librustc/ty/sty.rs @@ -1019,7 +1019,7 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { /// visible. pub fn is_uninhabited_from(&self, module: DefId, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> bool { let mut visited = FxHashSet::default(); - let forest = self.uninhabited_from(&mut visited, tcx); + let forest = self.uninhabited_from(&mut visited, 0, tcx); // To check whether this type is uninhabited at all (not just from the // given node) you could check whether the forest is empty. diff --git a/src/librustc_const_eval/_match.rs b/src/librustc_const_eval/_match.rs index 7a64ff7114a7e..101c43332a326 100644 --- a/src/librustc_const_eval/_match.rs +++ b/src/librustc_const_eval/_match.rs @@ -405,7 +405,7 @@ fn all_constructors<'a, 'tcx: 'a>(cx: &mut MatchCheckCtxt<'a, 'tcx>, ty::TyAdt(def, substs) if def.is_enum() && def.variants.len() != 1 => { def.variants.iter().filter_map(|v| { let mut visited = FxHashSet::default(); - let forest = v.uninhabited_from(&mut visited, + let forest = v.uninhabited_from(&mut visited, 0, cx.tcx, substs, AdtKind::Enum); if forest.contains(cx.tcx, cx.module) diff --git a/src/librustc_mir/build/matches/simplify.rs b/src/librustc_mir/build/matches/simplify.rs index e94d35195c213..f6bfed8c7ab5c 100644 --- a/src/librustc_mir/build/matches/simplify.rs +++ b/src/librustc_mir/build/matches/simplify.rs @@ -103,7 +103,7 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> { let irrefutable = adt_def.variants.iter().enumerate().all(|(i, v)| { i == variant_index || { let mut visited = FxHashSet::default(); - let node_set = v.uninhabited_from(&mut visited, + let node_set = v.uninhabited_from(&mut visited, 0, self.hir.tcx(), substs, adt_def.adt_kind()); diff --git a/src/test/compile-fail/inhabitedness-infinite-loop.rs b/src/test/compile-fail/inhabitedness-infinite-loop.rs new file mode 100644 index 0000000000000..eba0da3a216aa --- /dev/null +++ b/src/test/compile-fail/inhabitedness-infinite-loop.rs @@ -0,0 +1,24 @@ +// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT +// file at the top-level directory of this distribution and at +// http://rust-lang.org/COPYRIGHT. +// +// Licensed under the Apache License, Version 2.0 or the MIT license +// , at your +// option. This file may not be copied, modified, or distributed +// except according to those terms. + +#![feature(never_type)] + +struct Foo<'a, T: 'a> { + ph: std::marker::PhantomData, + foo: &'a Foo<'a, (T, T)>, +} + +fn wub(f: Foo) { + match f {} + //~^ ERROR non-exhaustive +} + +fn main() {} + From 347bc77c2c01b1cc78d1d4c3ea4cb48f196cee04 Mon Sep 17 00:00:00 2001 From: Andrew Cann Date: Fri, 10 Feb 2017 00:52:51 +0800 Subject: [PATCH 2/2] Use global recursion limit when evaluating inhabitedness --- src/librustc/ty/inhabitedness/mod.rs | 81 ++++++++++--------- src/librustc/ty/sty.rs | 6 +- src/librustc_const_eval/_match.rs | 6 +- src/librustc_mir/build/matches/simplify.rs | 6 +- .../inhabitedness-infinite-loop.rs | 3 +- 5 files changed, 55 insertions(+), 47 deletions(-) diff --git a/src/librustc/ty/inhabitedness/mod.rs b/src/librustc/ty/inhabitedness/mod.rs index 51d046728c2bf..24ca476e5ff79 100644 --- a/src/librustc/ty/inhabitedness/mod.rs +++ b/src/librustc/ty/inhabitedness/mod.rs @@ -8,7 +8,7 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. -use util::nodemap::FxHashSet; +use util::nodemap::{FxHashMap, FxHashSet}; use ty::context::TyCtxt; use ty::{AdtDef, VariantDef, FieldDef, TyS}; use ty::{DefId, Substs}; @@ -62,26 +62,17 @@ mod def_id_forest; // This code should only compile in modules where the uninhabitedness of Foo is // visible. -const ARBITRARY_RECURSION_LIMIT: u32 = 24; - impl<'a, 'gcx, 'tcx> AdtDef { /// Calculate the forest of DefIds from which this adt is visibly uninhabited. pub fn uninhabited_from( &self, - visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, - recursion_depth: u32, + visited: &mut FxHashMap>>, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>) -> DefIdForest { - if !visited.insert((self.did, substs)) { - return DefIdForest::empty(); - } - - let ret = DefIdForest::intersection(tcx, self.variants.iter().map(|v| { - v.uninhabited_from(visited, recursion_depth, tcx, substs, self.adt_kind()) - })); - visited.remove(&(self.did, substs)); - ret + DefIdForest::intersection(tcx, self.variants.iter().map(|v| { + v.uninhabited_from(visited, tcx, substs, self.adt_kind()) + })) } } @@ -89,8 +80,7 @@ impl<'a, 'gcx, 'tcx> VariantDef { /// Calculate the forest of DefIds from which this variant is visibly uninhabited. pub fn uninhabited_from( &self, - visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, - recursion_depth: u32, + visited: &mut FxHashMap>>, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>, adt_kind: AdtKind) -> DefIdForest @@ -98,17 +88,17 @@ impl<'a, 'gcx, 'tcx> VariantDef { match adt_kind { AdtKind::Union => { DefIdForest::intersection(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, recursion_depth, tcx, substs, false) + f.uninhabited_from(visited, tcx, substs, false) })) }, AdtKind::Struct => { DefIdForest::union(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, recursion_depth, tcx, substs, false) + f.uninhabited_from(visited, tcx, substs, false) })) }, AdtKind::Enum => { DefIdForest::union(tcx, self.fields.iter().map(|f| { - f.uninhabited_from(visited, recursion_depth, tcx, substs, true) + f.uninhabited_from(visited, tcx, substs, true) })) }, } @@ -119,14 +109,13 @@ impl<'a, 'gcx, 'tcx> FieldDef { /// Calculate the forest of DefIds from which this field is visibly uninhabited. pub fn uninhabited_from( &self, - visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, - recursion_depth: u32, + visited: &mut FxHashMap>>, tcx: TyCtxt<'a, 'gcx, 'tcx>, substs: &'tcx Substs<'tcx>, is_enum: bool) -> DefIdForest { let mut data_uninhabitedness = move || { - self.ty(tcx, substs).uninhabited_from(visited, recursion_depth, tcx) + self.ty(tcx, substs).uninhabited_from(visited, tcx) }; // FIXME(canndrew): Currently enum fields are (incorrectly) stored with // Visibility::Invisible so we need to override self.vis if we're @@ -151,15 +140,9 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { /// Calculate the forest of DefIds from which this type is visibly uninhabited. pub fn uninhabited_from( &self, - visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, - mut recursion_depth: u32, + visited: &mut FxHashMap>>, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> DefIdForest { - recursion_depth += 1; - if recursion_depth >= ARBITRARY_RECURSION_LIMIT { - return DefIdForest::empty(); - } - match tcx.lift_to_global(&self) { Some(global_ty) => { { @@ -168,13 +151,13 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { return forest.clone(); } } - let forest = global_ty.uninhabited_from_inner(visited, recursion_depth, tcx); + let forest = global_ty.uninhabited_from_inner(visited, tcx); let mut cache = tcx.inhabitedness_cache.borrow_mut(); cache.insert(global_ty, forest.clone()); forest }, None => { - let forest = self.uninhabited_from_inner(visited, recursion_depth, tcx); + let forest = self.uninhabited_from_inner(visited, tcx); forest }, } @@ -182,30 +165,54 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { fn uninhabited_from_inner( &self, - visited: &mut FxHashSet<(DefId, &'tcx Substs<'tcx>)>, - recursion_depth: u32, + visited: &mut FxHashMap>>, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> DefIdForest { match self.sty { TyAdt(def, substs) => { - def.uninhabited_from(visited, recursion_depth, tcx, substs) + { + let mut substs_set = visited.entry(def.did).or_insert(FxHashSet::default()); + if !substs_set.insert(substs) { + // We are already calculating the inhabitedness of this type. + // The type must contain a reference to itself. Break the + // infinite loop. + return DefIdForest::empty(); + } + if substs_set.len() >= tcx.sess.recursion_limit.get() / 4 { + // We have gone very deep, reinstantiating this ADT inside + // itself with different type arguments. We are probably + // hitting an infinite loop. For example, it's possible to write: + // a type Foo + // which contains a Foo<(T, T)> + // which contains a Foo<((T, T), (T, T))> + // which contains a Foo<(((T, T), (T, T)), ((T, T), (T, T)))> + // etc. + let error = format!("reached recursion limit while checking + inhabitedness of `{}`", self); + tcx.sess.fatal(&error); + } + } + let ret = def.uninhabited_from(visited, tcx, substs); + let mut substs_set = visited.get_mut(&def.did).unwrap(); + substs_set.remove(substs); + ret }, TyNever => DefIdForest::full(tcx), TyTuple(ref tys, _) => { DefIdForest::union(tcx, tys.iter().map(|ty| { - ty.uninhabited_from(visited, recursion_depth, tcx) + ty.uninhabited_from(visited, tcx) })) }, TyArray(ty, len) => { if len == 0 { DefIdForest::empty() } else { - ty.uninhabited_from(visited, recursion_depth, tcx) + ty.uninhabited_from(visited, tcx) } } TyRef(_, ref tm) => { - tm.ty.uninhabited_from(visited, recursion_depth, tcx) + tm.ty.uninhabited_from(visited, tcx) } _ => DefIdForest::empty(), diff --git a/src/librustc/ty/sty.rs b/src/librustc/ty/sty.rs index 61b774924260d..862bc15c05260 100644 --- a/src/librustc/ty/sty.rs +++ b/src/librustc/ty/sty.rs @@ -24,7 +24,7 @@ use std::cmp::Ordering; use syntax::abi; use syntax::ast::{self, Name}; use syntax::symbol::{keywords, InternedString}; -use util::nodemap::FxHashSet; +use util::nodemap::FxHashMap; use serialize; @@ -1018,8 +1018,8 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> { /// This code should only compile in modules where the uninhabitedness of Foo is /// visible. pub fn is_uninhabited_from(&self, module: DefId, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> bool { - let mut visited = FxHashSet::default(); - let forest = self.uninhabited_from(&mut visited, 0, tcx); + let mut visited = FxHashMap::default(); + let forest = self.uninhabited_from(&mut visited, tcx); // To check whether this type is uninhabited at all (not just from the // given node) you could check whether the forest is empty. diff --git a/src/librustc_const_eval/_match.rs b/src/librustc_const_eval/_match.rs index 101c43332a326..78c4027aa4319 100644 --- a/src/librustc_const_eval/_match.rs +++ b/src/librustc_const_eval/_match.rs @@ -17,7 +17,7 @@ use eval::{compare_const_vals}; use rustc_const_math::ConstInt; -use rustc_data_structures::fx::{FxHashMap, FxHashSet}; +use rustc_data_structures::fx::FxHashMap; use rustc_data_structures::indexed_vec::Idx; use pattern::{FieldPattern, Pattern, PatternKind}; @@ -404,8 +404,8 @@ fn all_constructors<'a, 'tcx: 'a>(cx: &mut MatchCheckCtxt<'a, 'tcx>, } ty::TyAdt(def, substs) if def.is_enum() && def.variants.len() != 1 => { def.variants.iter().filter_map(|v| { - let mut visited = FxHashSet::default(); - let forest = v.uninhabited_from(&mut visited, 0, + let mut visited = FxHashMap::default(); + let forest = v.uninhabited_from(&mut visited, cx.tcx, substs, AdtKind::Enum); if forest.contains(cx.tcx, cx.module) diff --git a/src/librustc_mir/build/matches/simplify.rs b/src/librustc_mir/build/matches/simplify.rs index f6bfed8c7ab5c..efddee2c933f4 100644 --- a/src/librustc_mir/build/matches/simplify.rs +++ b/src/librustc_mir/build/matches/simplify.rs @@ -26,7 +26,7 @@ use build::{BlockAnd, BlockAndExtension, Builder}; use build::matches::{Binding, MatchPair, Candidate}; use hair::*; use rustc::mir::*; -use rustc_data_structures::fx::FxHashSet; +use rustc_data_structures::fx::FxHashMap; use std::mem; @@ -102,8 +102,8 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> { if self.hir.tcx().sess.features.borrow().never_type { let irrefutable = adt_def.variants.iter().enumerate().all(|(i, v)| { i == variant_index || { - let mut visited = FxHashSet::default(); - let node_set = v.uninhabited_from(&mut visited, 0, + let mut visited = FxHashMap::default(); + let node_set = v.uninhabited_from(&mut visited, self.hir.tcx(), substs, adt_def.adt_kind()); diff --git a/src/test/compile-fail/inhabitedness-infinite-loop.rs b/src/test/compile-fail/inhabitedness-infinite-loop.rs index eba0da3a216aa..91b85d7510a24 100644 --- a/src/test/compile-fail/inhabitedness-infinite-loop.rs +++ b/src/test/compile-fail/inhabitedness-infinite-loop.rs @@ -8,6 +8,8 @@ // option. This file may not be copied, modified, or distributed // except according to those terms. +// error-pattern:reached recursion limit + #![feature(never_type)] struct Foo<'a, T: 'a> { @@ -17,7 +19,6 @@ struct Foo<'a, T: 'a> { fn wub(f: Foo) { match f {} - //~^ ERROR non-exhaustive } fn main() {}