Skip to content

Commit 5a2bec9

Browse files
committed
impl_or_trait_obligations: deduplicate obligations
1 parent 619ad71 commit 5a2bec9

File tree

3 files changed

+14
-7
lines changed

3 files changed

+14
-7
lines changed

src/librustc/traits/mod.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ pub enum IntercrateMode {
7373
/// either identifying an `impl` (e.g., `impl Eq for int`) that
7474
/// provides the required vtable, or else finding a bound that is in
7575
/// scope. The eventual result is usually a `Selection` (defined below).
76-
#[derive(Clone, PartialEq, Eq)]
76+
#[derive(Clone, PartialEq, Eq, Hash)]
7777
pub struct Obligation<'tcx, T> {
7878
pub cause: ObligationCause<'tcx>,
7979
pub param_env: ty::ParamEnv<'tcx>,
@@ -85,7 +85,7 @@ pub type PredicateObligation<'tcx> = Obligation<'tcx, ty::Predicate<'tcx>>;
8585
pub type TraitObligation<'tcx> = Obligation<'tcx, ty::PolyTraitPredicate<'tcx>>;
8686

8787
/// Why did we incur this obligation? Used for error reporting.
88-
#[derive(Clone, Debug, PartialEq, Eq)]
88+
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
8989
pub struct ObligationCause<'tcx> {
9090
pub span: Span,
9191

@@ -113,7 +113,7 @@ impl<'tcx> ObligationCause<'tcx> {
113113
}
114114
}
115115

116-
#[derive(Clone, Debug, PartialEq, Eq)]
116+
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
117117
pub enum ObligationCauseCode<'tcx> {
118118
/// Not well classified or should be obvious from span.
119119
MiscObligation,
@@ -215,7 +215,7 @@ pub enum ObligationCauseCode<'tcx> {
215215
BlockTailExpression(ast::NodeId),
216216
}
217217

218-
#[derive(Clone, Debug, PartialEq, Eq)]
218+
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
219219
pub struct DerivedObligationCause<'tcx> {
220220
/// The trait reference of the parent obligation that led to the
221221
/// current obligation. Note that only trait obligations lead to

src/librustc/traits/select.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ use std::rc::Rc;
5353
use syntax::abi::Abi;
5454
use hir;
5555
use lint;
56-
use util::nodemap::FxHashMap;
56+
use util::nodemap::{FxHashMap, FxHashSet};
5757

5858
struct InferredObligationsSnapshotVecDelegate<'tcx> {
5959
phantom: PhantomData<&'tcx i32>,
@@ -3282,7 +3282,7 @@ impl<'cx, 'gcx, 'tcx> SelectionContext<'cx, 'gcx, 'tcx> {
32823282
// that order.
32833283
let predicates = tcx.predicates_of(def_id);
32843284
assert_eq!(predicates.parent, None);
3285-
let predicates = predicates.predicates.iter().flat_map(|predicate| {
3285+
let mut predicates: Vec<_> = predicates.predicates.iter().flat_map(|predicate| {
32863286
let predicate = normalize_with_depth(self, param_env, cause.clone(), recursion_depth,
32873287
&predicate.subst(tcx, substs));
32883288
predicate.obligations.into_iter().chain(
@@ -3293,6 +3293,13 @@ impl<'cx, 'gcx, 'tcx> SelectionContext<'cx, 'gcx, 'tcx> {
32933293
predicate: predicate.value
32943294
}))
32953295
}).collect();
3296+
// We are performing deduplication here to avoid exponential blowups
3297+
// (#38528) from happening, but the real cause of the duplication is
3298+
// unknown. What we know is that the deduplication avoids exponential
3299+
// amount of predicates being propogated when processing deeply nested
3300+
// types.
3301+
let mut seen = FxHashSet();
3302+
predicates.retain(|i| seen.insert(i.clone()));
32963303
self.infcx().plug_leaks(skol_map, snapshot, predicates)
32973304
}
32983305
}

src/librustc/ty/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1504,7 +1504,7 @@ impl<'gcx> HashStable<StableHashingContext<'gcx>> for AdtDef {
15041504
}
15051505
}
15061506

1507-
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
1507+
#[derive(Copy, Clone, Debug, Eq, PartialEq, Hash)]
15081508
pub enum AdtKind { Struct, Union, Enum }
15091509

15101510
bitflags! {

0 commit comments

Comments
 (0)