Skip to content

Commit 0956690

Browse files
Rollup merge of rust-lang#55330 - scalexm:bound-ty, r=nikomatsakis
Add support for bound types This PR may have some slight performance impacts, I don't know how hot is the code I touched. Also, this breaks clippy and miri. r? @nikomatsakis
2 parents 805fcb6 + 2b205dc commit 0956690

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

72 files changed

+554
-389
lines changed

src/librustc/ich/impls_ty.rs

+4-5
Original file line numberDiff line numberDiff line change
@@ -100,9 +100,6 @@ for ty::RegionKind {
100100
ty::ReEmpty => {
101101
// No variant fields to hash for these ...
102102
}
103-
ty::ReCanonical(c) => {
104-
c.hash_stable(hcx, hasher);
105-
}
106103
ty::ReLateBound(db, ty::BrAnon(i)) => {
107104
db.hash_stable(hcx, hasher);
108105
i.hash_stable(hcx, hasher);
@@ -147,7 +144,7 @@ impl<'a> HashStable<StableHashingContext<'a>> for ty::RegionVid {
147144
}
148145
}
149146

150-
impl<'gcx> HashStable<StableHashingContext<'gcx>> for ty::BoundTyIndex {
147+
impl<'gcx> HashStable<StableHashingContext<'gcx>> for ty::BoundVar {
151148
#[inline]
152149
fn hash_stable<W: StableHasherResult>(&self,
153150
hcx: &mut StableHashingContext<'gcx>,
@@ -894,6 +891,9 @@ for ty::TyKind<'gcx>
894891
Param(param_ty) => {
895892
param_ty.hash_stable(hcx, hasher);
896893
}
894+
Bound(bound_ty) => {
895+
bound_ty.hash_stable(hcx, hasher);
896+
}
897897
Foreign(def_id) => {
898898
def_id.hash_stable(hcx, hasher);
899899
}
@@ -911,7 +911,6 @@ impl_stable_hash_for!(enum ty::InferTy {
911911
FreshTy(a),
912912
FreshIntTy(a),
913913
FreshFloatTy(a),
914-
BoundTy(a),
915914
});
916915

917916
impl<'a, 'gcx> HashStable<StableHashingContext<'a>>

src/librustc/infer/canonical/canonicalizer.rs

+42-30
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ use infer::InferCtxt;
2323
use std::sync::atomic::Ordering;
2424
use ty::fold::{TypeFoldable, TypeFolder};
2525
use ty::subst::Kind;
26-
use ty::{self, BoundTy, BoundTyIndex, Lift, List, Ty, TyCtxt, TypeFlags};
26+
use ty::{self, BoundTy, BoundVar, Lift, List, Ty, TyCtxt, TypeFlags};
2727

2828
use rustc_data_structures::fx::FxHashMap;
2929
use rustc_data_structures::indexed_vec::Idx;
@@ -225,21 +225,35 @@ struct Canonicalizer<'cx, 'gcx: 'tcx, 'tcx: 'cx> {
225225
query_state: &'cx mut OriginalQueryValues<'tcx>,
226226
// Note that indices is only used once `var_values` is big enough to be
227227
// heap-allocated.
228-
indices: FxHashMap<Kind<'tcx>, BoundTyIndex>,
228+
indices: FxHashMap<Kind<'tcx>, BoundVar>,
229229
canonicalize_region_mode: &'cx dyn CanonicalizeRegionMode,
230230
needs_canonical_flags: TypeFlags,
231+
232+
binder_index: ty::DebruijnIndex,
231233
}
232234

233235
impl<'cx, 'gcx, 'tcx> TypeFolder<'gcx, 'tcx> for Canonicalizer<'cx, 'gcx, 'tcx> {
234236
fn tcx<'b>(&'b self) -> TyCtxt<'b, 'gcx, 'tcx> {
235237
self.tcx
236238
}
237239

240+
fn fold_binder<T>(&mut self, t: &ty::Binder<T>) -> ty::Binder<T>
241+
where T: TypeFoldable<'tcx>
242+
{
243+
self.binder_index.shift_in(1);
244+
let t = t.super_fold_with(self);
245+
self.binder_index.shift_out(1);
246+
t
247+
}
248+
238249
fn fold_region(&mut self, r: ty::Region<'tcx>) -> ty::Region<'tcx> {
239250
match *r {
240-
ty::ReLateBound(..) => {
241-
// leave bound regions alone
242-
r
251+
ty::ReLateBound(index, ..) => {
252+
if index >= self.binder_index {
253+
bug!("escaping late bound region during canonicalization")
254+
} else {
255+
r
256+
}
243257
}
244258

245259
ty::ReVar(vid) => {
@@ -263,8 +277,8 @@ impl<'cx, 'gcx, 'tcx> TypeFolder<'gcx, 'tcx> for Canonicalizer<'cx, 'gcx, 'tcx>
263277
| ty::ReEmpty
264278
| ty::ReErased => self.canonicalize_region_mode.canonicalize_free_region(self, r),
265279

266-
ty::ReClosureBound(..) | ty::ReCanonical(_) => {
267-
bug!("canonical region encountered during canonicalization")
280+
ty::ReClosureBound(..) => {
281+
bug!("closure bound region encountered during canonicalization")
268282
}
269283
}
270284
}
@@ -283,8 +297,12 @@ impl<'cx, 'gcx, 'tcx> TypeFolder<'gcx, 'tcx> for Canonicalizer<'cx, 'gcx, 'tcx>
283297
bug!("encountered a fresh type during canonicalization")
284298
}
285299

286-
ty::Infer(ty::BoundTy(_)) => {
287-
bug!("encountered a canonical type during canonicalization")
300+
ty::Bound(bound_ty) => {
301+
if bound_ty.index >= self.binder_index {
302+
bug!("escaping bound type during canonicalization")
303+
} else {
304+
t
305+
}
288306
}
289307

290308
ty::Closure(..)
@@ -335,12 +353,6 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
335353
where
336354
V: TypeFoldable<'tcx> + Lift<'gcx>,
337355
{
338-
debug_assert!(
339-
!value.has_type_flags(TypeFlags::HAS_CANONICAL_VARS),
340-
"canonicalizing a canonical value: {:?}",
341-
value,
342-
);
343-
344356
let needs_canonical_flags = if canonicalize_region_mode.any() {
345357
TypeFlags::HAS_FREE_REGIONS | TypeFlags::KEEP_IN_LOCAL_TCX
346358
} else {
@@ -367,6 +379,7 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
367379
variables: SmallVec::new(),
368380
query_state,
369381
indices: FxHashMap::default(),
382+
binder_index: ty::INNERMOST,
370383
};
371384
let out_value = value.fold_with(&mut canonicalizer);
372385

@@ -393,7 +406,7 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
393406
/// or returns an existing variable if `kind` has already been
394407
/// seen. `kind` is expected to be an unbound variable (or
395408
/// potentially a free region).
396-
fn canonical_var(&mut self, info: CanonicalVarInfo, kind: Kind<'tcx>) -> BoundTy {
409+
fn canonical_var(&mut self, info: CanonicalVarInfo, kind: Kind<'tcx>) -> BoundVar {
397410
let Canonicalizer {
398411
variables,
399412
query_state,
@@ -413,7 +426,7 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
413426
// direct linear search of `var_values`.
414427
if let Some(idx) = var_values.iter().position(|&k| k == kind) {
415428
// `kind` is already present in `var_values`.
416-
BoundTyIndex::new(idx)
429+
BoundVar::new(idx)
417430
} else {
418431
// `kind` isn't present in `var_values`. Append it. Likewise
419432
// for `info` and `variables`.
@@ -428,35 +441,35 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
428441
*indices = var_values
429442
.iter()
430443
.enumerate()
431-
.map(|(i, &kind)| (kind, BoundTyIndex::new(i)))
444+
.map(|(i, &kind)| (kind, BoundVar::new(i)))
432445
.collect();
433446
}
434447
// The cv is the index of the appended element.
435-
BoundTyIndex::new(var_values.len() - 1)
448+
BoundVar::new(var_values.len() - 1)
436449
}
437450
} else {
438451
// `var_values` is large. Do a hashmap search via `indices`.
439452
*indices.entry(kind).or_insert_with(|| {
440453
variables.push(info);
441454
var_values.push(kind);
442455
assert_eq!(variables.len(), var_values.len());
443-
BoundTyIndex::new(variables.len() - 1)
456+
BoundVar::new(variables.len() - 1)
444457
})
445458
};
446459

447-
BoundTy {
448-
level: ty::INNERMOST,
449-
var,
450-
}
460+
var
451461
}
452462

453463
fn canonical_var_for_region(&mut self, r: ty::Region<'tcx>) -> ty::Region<'tcx> {
454464
let info = CanonicalVarInfo {
455465
kind: CanonicalVarKind::Region,
456466
};
457-
let b = self.canonical_var(info, r.into());
458-
debug_assert_eq!(ty::INNERMOST, b.level);
459-
self.tcx().mk_region(ty::ReCanonical(b.var))
467+
let var = self.canonical_var(info, r.into());
468+
let region = ty::ReLateBound(
469+
self.binder_index,
470+
ty::BoundRegion::BrAnon(var.as_u32())
471+
);
472+
self.tcx().mk_region(region)
460473
}
461474

462475
/// Given a type variable `ty_var` of the given kind, first check
@@ -472,9 +485,8 @@ impl<'cx, 'gcx, 'tcx> Canonicalizer<'cx, 'gcx, 'tcx> {
472485
let info = CanonicalVarInfo {
473486
kind: CanonicalVarKind::Ty(ty_kind),
474487
};
475-
let b = self.canonical_var(info, ty_var.into());
476-
debug_assert_eq!(ty::INNERMOST, b.level);
477-
self.tcx().mk_infer(ty::InferTy::BoundTy(b))
488+
let var = self.canonical_var(info, ty_var.into());
489+
self.tcx().mk_ty(ty::Bound(BoundTy::new(self.binder_index, var)))
478490
}
479491
}
480492
}

src/librustc/infer/canonical/mod.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020
//! - a map M (of type `CanonicalVarValues`) from those canonical
2121
//! variables back to the original.
2222
//!
23-
//! We can then do queries using T2. These will give back constriants
23+
//! We can then do queries using T2. These will give back constraints
2424
//! on the canonical variables which can be translated, using the map
2525
//! M, into constraints in our source context. This process of
2626
//! translating the results back is done by the
@@ -40,7 +40,7 @@ use std::ops::Index;
4040
use syntax::source_map::Span;
4141
use ty::fold::TypeFoldable;
4242
use ty::subst::Kind;
43-
use ty::{self, BoundTyIndex, Lift, Region, List, TyCtxt};
43+
use ty::{self, BoundVar, Lift, Region, List, TyCtxt};
4444

4545
mod canonicalizer;
4646

@@ -72,7 +72,7 @@ impl<'gcx> UseSpecializedDecodable for CanonicalVarInfos<'gcx> {}
7272
/// canonicalized query response.
7373
#[derive(Clone, Debug, PartialEq, Eq, Hash, RustcDecodable, RustcEncodable)]
7474
pub struct CanonicalVarValues<'tcx> {
75-
pub var_values: IndexVec<BoundTyIndex, Kind<'tcx>>,
75+
pub var_values: IndexVec<BoundVar, Kind<'tcx>>,
7676
}
7777

7878
/// When we canonicalize a value to form a query, we wind up replacing
@@ -264,7 +264,7 @@ impl<'cx, 'gcx, 'tcx> InferCtxt<'cx, 'gcx, 'tcx> {
264264
span: Span,
265265
variables: &List<CanonicalVarInfo>,
266266
) -> CanonicalVarValues<'tcx> {
267-
let var_values: IndexVec<BoundTyIndex, Kind<'tcx>> = variables
267+
let var_values: IndexVec<BoundVar, Kind<'tcx>> = variables
268268
.iter()
269269
.map(|info| self.fresh_inference_var_for_canonical_var(span, *info))
270270
.collect();
@@ -367,10 +367,10 @@ BraceStructLiftImpl! {
367367
} where R: Lift<'tcx>
368368
}
369369

370-
impl<'tcx> Index<BoundTyIndex> for CanonicalVarValues<'tcx> {
370+
impl<'tcx> Index<BoundVar> for CanonicalVarValues<'tcx> {
371371
type Output = Kind<'tcx>;
372372

373-
fn index(&self, value: BoundTyIndex) -> &Kind<'tcx> {
373+
fn index(&self, value: BoundVar) -> &Kind<'tcx> {
374374
&self.var_values[value]
375375
}
376376
}

0 commit comments

Comments
 (0)