2013-10-29 04:25:18 -05:00
|
|
|
// Copyright 2012-2013 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 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2014-11-25 20:17:11 -06:00
|
|
|
//! Generalized type folding mechanism. The setup is a bit convoluted
|
|
|
|
//! but allows for convenient usage. Let T be an instance of some
|
|
|
|
//! "foldable type" (one which implements `TypeFoldable`) and F be an
|
|
|
|
//! instance of a "folder" (a type which implements `TypeFolder`). Then
|
|
|
|
//! the setup is intended to be:
|
|
|
|
//!
|
|
|
|
//! T.fold_with(F) --calls--> F.fold_T(T) --calls--> super_fold_T(F, T)
|
|
|
|
//!
|
|
|
|
//! This way, when you define a new folder F, you can override
|
|
|
|
//! `fold_T()` to customize the behavior, and invoke `super_fold_T()`
|
|
|
|
//! to get the original behavior. Meanwhile, to actually fold
|
|
|
|
//! something, you can just write `T.fold_with(F)`, which is
|
|
|
|
//! convenient. (Note that `fold_with` will also transparently handle
|
|
|
|
//! things like a `Vec<T>` where T is foldable and so on.)
|
|
|
|
//!
|
|
|
|
//! In this ideal setup, the only function that actually *does*
|
|
|
|
//! anything is `super_fold_T`, which traverses the type `T`. Moreover,
|
|
|
|
//! `super_fold_T` should only ever call `T.fold_with()`.
|
|
|
|
//!
|
|
|
|
//! In some cases, we follow a degenerate pattern where we do not have
|
|
|
|
//! a `fold_T` nor `super_fold_T` method. Instead, `T.fold_with`
|
|
|
|
//! traverses the structure directly. This is suboptimal because the
|
2015-01-06 19:53:18 -06:00
|
|
|
//! behavior cannot be overridden, but it's much less work to implement.
|
2014-11-25 20:17:11 -06:00
|
|
|
//! If you ever *do* need an override that doesn't exist, it's not hard
|
|
|
|
//! to convert the degenerate pattern into the proper thing.
|
2013-10-29 04:25:18 -05:00
|
|
|
|
2015-09-06 13:51:58 -05:00
|
|
|
use middle::region;
|
2014-05-13 10:35:42 -05:00
|
|
|
use middle::subst;
|
2015-09-14 06:55:56 -05:00
|
|
|
use middle::ty::adjustment;
|
2015-09-14 16:47:14 -05:00
|
|
|
use middle::ty::{self, Binder, Ty, RegionEscape};
|
2015-06-18 00:51:23 -05:00
|
|
|
|
|
|
|
use std::fmt;
|
2015-06-29 16:32:39 -05:00
|
|
|
use util::nodemap::{FnvHashMap, FnvHashSet};
|
2013-10-29 04:25:18 -05:00
|
|
|
|
2014-05-12 16:12:51 -05:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Two generic traits
|
|
|
|
|
|
|
|
/// The TypeFoldable trait is implemented for every type that can be folded.
|
|
|
|
/// Basically, every type that has a corresponding method in TypeFolder.
|
2015-06-18 00:51:23 -05:00
|
|
|
pub trait TypeFoldable<'tcx>: fmt::Debug + Clone {
|
2014-09-29 14:11:30 -05:00
|
|
|
fn fold_with<F: TypeFolder<'tcx>>(&self, folder: &mut F) -> Self;
|
2014-05-12 16:12:51 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// The TypeFolder trait defines the actual *folding*. There is a
|
|
|
|
/// method defined for every foldable type. Each of these has a
|
|
|
|
/// default implementation that does an "identity" fold. Within each
|
|
|
|
/// identity fold, it should invoke `foo.fold_with(self)` to fold each
|
|
|
|
/// sub-item.
|
2014-12-18 14:27:41 -06:00
|
|
|
pub trait TypeFolder<'tcx> : Sized {
|
2014-04-22 07:56:37 -05:00
|
|
|
fn tcx<'a>(&'a self) -> &'a ty::ctxt<'tcx>;
|
2013-10-29 04:25:18 -05:00
|
|
|
|
2014-11-15 15:47:59 -06:00
|
|
|
/// Invoked by the `super_*` routines when we enter a region
|
|
|
|
/// binding level (for example, when entering a function
|
|
|
|
/// signature). This is used by clients that want to track the
|
|
|
|
/// Debruijn index nesting level.
|
|
|
|
fn enter_region_binder(&mut self) { }
|
|
|
|
|
|
|
|
/// Invoked by the `super_*` routines when we exit a region
|
|
|
|
/// binding level. This is used by clients that want to
|
|
|
|
/// track the Debruijn index nesting level.
|
|
|
|
fn exit_region_binder(&mut self) { }
|
|
|
|
|
2015-09-06 13:51:58 -05:00
|
|
|
fn fold_binder<T>(&mut self, t: &Binder<T>) -> Binder<T>
|
2015-06-18 00:51:23 -05:00
|
|
|
where T : TypeFoldable<'tcx>
|
2015-01-04 05:07:36 -06:00
|
|
|
{
|
2015-01-04 11:00:13 -06:00
|
|
|
// FIXME(#20526) this should replace `enter_region_binder`/`exit_region_binder`.
|
2015-01-04 05:07:36 -06:00
|
|
|
super_fold_binder(self, t)
|
|
|
|
}
|
|
|
|
|
2014-09-29 14:11:30 -05:00
|
|
|
fn fold_ty(&mut self, t: Ty<'tcx>) -> Ty<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
super_fold_ty(self, t)
|
|
|
|
}
|
|
|
|
|
2015-07-10 20:27:06 -05:00
|
|
|
fn fold_mt(&mut self, t: &ty::TypeAndMut<'tcx>) -> ty::TypeAndMut<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
super_fold_mt(self, t)
|
|
|
|
}
|
|
|
|
|
2014-09-29 14:11:30 -05:00
|
|
|
fn fold_trait_ref(&mut self, t: &ty::TraitRef<'tcx>) -> ty::TraitRef<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
super_fold_trait_ref(self, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_substs(&mut self,
|
2014-09-29 14:11:30 -05:00
|
|
|
substs: &subst::Substs<'tcx>)
|
|
|
|
-> subst::Substs<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
super_fold_substs(self, substs)
|
|
|
|
}
|
|
|
|
|
2014-11-15 15:47:59 -06:00
|
|
|
fn fold_fn_sig(&mut self,
|
2014-12-12 10:28:35 -06:00
|
|
|
sig: &ty::FnSig<'tcx>)
|
|
|
|
-> ty::FnSig<'tcx> {
|
2014-11-15 15:47:59 -06:00
|
|
|
super_fold_fn_sig(self, sig)
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
2014-10-24 14:14:37 -05:00
|
|
|
fn fold_output(&mut self,
|
2014-09-29 14:11:30 -05:00
|
|
|
output: &ty::FnOutput<'tcx>)
|
|
|
|
-> ty::FnOutput<'tcx> {
|
2014-10-24 14:14:37 -05:00
|
|
|
super_fold_output(self, output)
|
|
|
|
}
|
|
|
|
|
2013-10-29 04:25:18 -05:00
|
|
|
fn fold_bare_fn_ty(&mut self,
|
2014-09-29 14:11:30 -05:00
|
|
|
fty: &ty::BareFnTy<'tcx>)
|
|
|
|
-> ty::BareFnTy<'tcx>
|
2014-05-12 16:12:51 -05:00
|
|
|
{
|
|
|
|
super_fold_bare_fn_ty(self, fty)
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_closure_ty(&mut self,
|
2014-09-29 14:11:30 -05:00
|
|
|
fty: &ty::ClosureTy<'tcx>)
|
|
|
|
-> ty::ClosureTy<'tcx> {
|
2014-05-12 16:12:51 -05:00
|
|
|
super_fold_closure_ty(self, fty)
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
2014-12-26 03:36:04 -06:00
|
|
|
fn fold_existential_bounds(&mut self, s: &ty::ExistentialBounds<'tcx>)
|
|
|
|
-> ty::ExistentialBounds<'tcx> {
|
2014-08-27 20:46:52 -05:00
|
|
|
super_fold_existential_bounds(self, s)
|
|
|
|
}
|
|
|
|
|
2015-09-14 06:55:56 -05:00
|
|
|
fn fold_autoref(&mut self, ar: &adjustment::AutoRef<'tcx>)
|
|
|
|
-> adjustment::AutoRef<'tcx> {
|
2014-05-06 14:16:11 -05:00
|
|
|
super_fold_autoref(self, ar)
|
|
|
|
}
|
2014-05-12 16:12:51 -05:00
|
|
|
|
2014-09-29 14:11:30 -05:00
|
|
|
fn fold_item_substs(&mut self, i: ty::ItemSubsts<'tcx>) -> ty::ItemSubsts<'tcx> {
|
2014-05-12 16:12:51 -05:00
|
|
|
super_fold_item_substs(self, i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// "super" routines: these are the default implementations for TypeFolder.
|
|
|
|
//
|
|
|
|
// They should invoke `foo.fold_with()` to do recursive folding.
|
2014-11-15 16:09:51 -06:00
|
|
|
|
2015-01-04 05:07:36 -06:00
|
|
|
pub fn super_fold_binder<'tcx, T, U>(this: &mut T,
|
2015-09-06 13:51:58 -05:00
|
|
|
binder: &Binder<U>)
|
|
|
|
-> Binder<U>
|
2015-01-04 05:07:36 -06:00
|
|
|
where T : TypeFolder<'tcx>, U : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
this.enter_region_binder();
|
2015-09-06 13:51:58 -05:00
|
|
|
let result = Binder(binder.0.fold_with(this));
|
2015-01-04 05:07:36 -06:00
|
|
|
this.exit_region_binder();
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-12-16 10:38:06 -06:00
|
|
|
ty: Ty<'tcx>)
|
2014-09-29 14:11:30 -05:00
|
|
|
-> Ty<'tcx> {
|
2014-12-16 10:38:06 -06:00
|
|
|
let sty = match ty.sty {
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyBox(typ) => {
|
|
|
|
ty::TyBox(typ.fold_with(this))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyRawPtr(ref tm) => {
|
|
|
|
ty::TyRawPtr(tm.fold_with(this))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyArray(typ, sz) => {
|
|
|
|
ty::TyArray(typ.fold_with(this), sz)
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-12 18:50:13 -05:00
|
|
|
ty::TySlice(typ) => {
|
|
|
|
ty::TySlice(typ.fold_with(this))
|
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyEnum(tid, ref substs) => {
|
2014-12-26 05:33:56 -06:00
|
|
|
let substs = substs.fold_with(this);
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyEnum(tid, this.tcx().mk_substs(substs))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyTrait(box ty::TraitTy { ref principal, ref bounds }) => {
|
|
|
|
ty::TyTrait(box ty::TraitTy {
|
2014-12-26 03:36:04 -06:00
|
|
|
principal: principal.fold_with(this),
|
2014-12-16 10:38:06 -06:00
|
|
|
bounds: bounds.fold_with(this),
|
|
|
|
})
|
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyTuple(ref ts) => {
|
|
|
|
ty::TyTuple(ts.fold_with(this))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyBareFn(opt_def_id, ref f) => {
|
2014-12-26 05:33:56 -06:00
|
|
|
let bfn = f.fold_with(this);
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyBareFn(opt_def_id, this.tcx().mk_bare_fn(bfn))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyRef(r, ref tm) => {
|
2014-12-26 05:33:56 -06:00
|
|
|
let r = r.fold_with(this);
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyRef(this.tcx().mk_region(r), tm.fold_with(this))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyStruct(did, ref substs) => {
|
2014-12-26 05:33:56 -06:00
|
|
|
let substs = substs.fold_with(this);
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyStruct(did, this.tcx().mk_substs(substs))
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-07-16 08:46:35 -05:00
|
|
|
ty::TyClosure(did, ref substs) => {
|
2014-12-26 05:33:56 -06:00
|
|
|
let s = substs.fold_with(this);
|
2015-07-16 08:46:35 -05:00
|
|
|
ty::TyClosure(did, s)
|
2014-12-16 10:38:06 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyProjection(ref data) => {
|
|
|
|
ty::TyProjection(data.fold_with(this))
|
2014-12-17 13:16:28 -06:00
|
|
|
}
|
2015-06-11 18:21:46 -05:00
|
|
|
ty::TyBool | ty::TyChar | ty::TyStr |
|
|
|
|
ty::TyInt(_) | ty::TyUint(_) | ty::TyFloat(_) |
|
|
|
|
ty::TyError | ty::TyInfer(_) |
|
|
|
|
ty::TyParam(..) => {
|
2014-12-16 10:38:06 -06:00
|
|
|
ty.sty.clone()
|
|
|
|
}
|
|
|
|
};
|
2015-06-24 20:09:46 -05:00
|
|
|
this.tcx().mk_ty(sty)
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_substs<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
substs: &subst::Substs<'tcx>)
|
|
|
|
-> subst::Substs<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
let regions = match substs.regions {
|
2014-05-13 10:35:42 -05:00
|
|
|
subst::ErasedRegions => {
|
|
|
|
subst::ErasedRegions
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
2014-05-13 10:35:42 -05:00
|
|
|
subst::NonerasedRegions(ref regions) => {
|
|
|
|
subst::NonerasedRegions(regions.fold_with(this))
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-05-13 10:35:42 -05:00
|
|
|
subst::Substs { regions: regions,
|
2014-05-31 17:53:13 -05:00
|
|
|
types: substs.types.fold_with(this) }
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
2014-11-15 15:47:59 -06:00
|
|
|
pub fn super_fold_fn_sig<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
sig: &ty::FnSig<'tcx>)
|
|
|
|
-> ty::FnSig<'tcx>
|
2014-11-15 15:47:59 -06:00
|
|
|
{
|
|
|
|
ty::FnSig { inputs: sig.inputs.fold_with(this),
|
2014-05-12 16:12:51 -05:00
|
|
|
output: sig.output.fold_with(this),
|
2013-10-29 04:25:18 -05:00
|
|
|
variadic: sig.variadic }
|
|
|
|
}
|
|
|
|
|
2014-10-24 14:14:37 -05:00
|
|
|
pub fn super_fold_output<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
output: &ty::FnOutput<'tcx>)
|
|
|
|
-> ty::FnOutput<'tcx> {
|
2014-10-24 14:14:37 -05:00
|
|
|
match *output {
|
|
|
|
ty::FnConverging(ref ty) => ty::FnConverging(ty.fold_with(this)),
|
|
|
|
ty::FnDiverging => ty::FnDiverging
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_bare_fn_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
fty: &ty::BareFnTy<'tcx>)
|
|
|
|
-> ty::BareFnTy<'tcx>
|
2014-05-12 16:12:51 -05:00
|
|
|
{
|
|
|
|
ty::BareFnTy { sig: fty.sig.fold_with(this),
|
|
|
|
abi: fty.abi,
|
2014-12-09 09:36:46 -06:00
|
|
|
unsafety: fty.unsafety }
|
2014-05-12 16:12:51 -05:00
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_closure_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
fty: &ty::ClosureTy<'tcx>)
|
|
|
|
-> ty::ClosureTy<'tcx>
|
2014-05-12 16:12:51 -05:00
|
|
|
{
|
|
|
|
ty::ClosureTy {
|
|
|
|
sig: fty.sig.fold_with(this),
|
2014-12-09 09:36:46 -06:00
|
|
|
unsafety: fty.unsafety,
|
2014-05-29 00:26:56 -05:00
|
|
|
abi: fty.abi,
|
2014-05-12 16:12:51 -05:00
|
|
|
}
|
|
|
|
}
|
2014-11-15 16:09:51 -06:00
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_trait_ref<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
t: &ty::TraitRef<'tcx>)
|
|
|
|
-> ty::TraitRef<'tcx>
|
2014-11-15 16:09:51 -06:00
|
|
|
{
|
2014-12-26 05:33:56 -06:00
|
|
|
let substs = t.substs.fold_with(this);
|
2013-10-29 04:25:18 -05:00
|
|
|
ty::TraitRef {
|
|
|
|
def_id: t.def_id,
|
2014-12-03 18:01:29 -06:00
|
|
|
substs: this.tcx().mk_substs(substs),
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_mt<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2015-07-10 20:27:06 -05:00
|
|
|
mt: &ty::TypeAndMut<'tcx>)
|
|
|
|
-> ty::TypeAndMut<'tcx> {
|
|
|
|
ty::TypeAndMut {ty: mt.ty.fold_with(this),
|
2013-10-29 04:25:18 -05:00
|
|
|
mutbl: mt.mutbl}
|
|
|
|
}
|
|
|
|
|
2014-12-26 03:36:04 -06:00
|
|
|
pub fn super_fold_existential_bounds<'tcx, T: TypeFolder<'tcx>>(
|
|
|
|
this: &mut T,
|
|
|
|
bounds: &ty::ExistentialBounds<'tcx>)
|
|
|
|
-> ty::ExistentialBounds<'tcx>
|
|
|
|
{
|
2014-08-27 20:46:52 -05:00
|
|
|
ty::ExistentialBounds {
|
|
|
|
region_bound: bounds.region_bound.fold_with(this),
|
|
|
|
builtin_bounds: bounds.builtin_bounds,
|
2014-12-26 03:36:04 -06:00
|
|
|
projection_bounds: bounds.projection_bounds.fold_with(this),
|
2014-08-27 20:46:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_autoref<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2015-09-14 06:55:56 -05:00
|
|
|
autoref: &adjustment::AutoRef<'tcx>)
|
|
|
|
-> adjustment::AutoRef<'tcx>
|
2014-05-06 14:16:11 -05:00
|
|
|
{
|
|
|
|
match *autoref {
|
2015-09-14 06:55:56 -05:00
|
|
|
adjustment::AutoPtr(r, m) => {
|
2015-03-16 11:45:01 -05:00
|
|
|
let r = r.fold_with(this);
|
2015-09-14 06:55:56 -05:00
|
|
|
adjustment::AutoPtr(this.tcx().mk_region(r), m)
|
2014-08-27 00:07:28 -05:00
|
|
|
}
|
2015-09-14 06:55:56 -05:00
|
|
|
adjustment::AutoUnsafe(m) => adjustment::AutoUnsafe(m)
|
2014-05-12 16:12:51 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 07:56:37 -05:00
|
|
|
pub fn super_fold_item_substs<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
2014-09-29 14:11:30 -05:00
|
|
|
substs: ty::ItemSubsts<'tcx>)
|
|
|
|
-> ty::ItemSubsts<'tcx>
|
2014-05-12 16:12:51 -05:00
|
|
|
{
|
|
|
|
ty::ItemSubsts {
|
|
|
|
substs: substs.substs.fold_with(this),
|
2014-05-06 14:16:11 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 16:09:51 -06:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
2013-10-29 04:25:18 -05:00
|
|
|
// Some sample folders
|
|
|
|
|
2014-12-08 19:26:43 -06:00
|
|
|
pub struct BottomUpFolder<'a, 'tcx: 'a, F> where F: FnMut(Ty<'tcx>) -> Ty<'tcx> {
|
2014-04-22 07:56:37 -05:00
|
|
|
pub tcx: &'a ty::ctxt<'tcx>,
|
2014-12-08 19:26:43 -06:00
|
|
|
pub fldop: F,
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
2014-12-08 19:26:43 -06:00
|
|
|
impl<'a, 'tcx, F> TypeFolder<'tcx> for BottomUpFolder<'a, 'tcx, F> where
|
|
|
|
F: FnMut(Ty<'tcx>) -> Ty<'tcx>,
|
|
|
|
{
|
2014-12-16 16:34:47 -06:00
|
|
|
fn tcx(&self) -> &ty::ctxt<'tcx> { self.tcx }
|
2013-10-29 04:25:18 -05:00
|
|
|
|
2014-09-29 14:11:30 -05:00
|
|
|
fn fold_ty(&mut self, ty: Ty<'tcx>) -> Ty<'tcx> {
|
2013-10-29 04:25:18 -05:00
|
|
|
let t1 = super_fold_ty(self, ty);
|
|
|
|
(self.fldop)(t1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Region folder
|
|
|
|
|
2015-09-06 13:51:58 -05:00
|
|
|
impl<'tcx> ty::ctxt<'tcx> {
|
|
|
|
/// Collects the free and escaping regions in `value` into `region_set`. Returns
|
|
|
|
/// whether any late-bound regions were skipped
|
|
|
|
pub fn collect_regions<T>(&self,
|
|
|
|
value: &T,
|
|
|
|
region_set: &mut FnvHashSet<ty::Region>)
|
|
|
|
-> bool
|
|
|
|
where T : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
let mut have_bound_regions = false;
|
|
|
|
self.fold_regions(value, &mut have_bound_regions,
|
|
|
|
|r, d| { region_set.insert(r.from_depth(d)); r });
|
|
|
|
have_bound_regions
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Folds the escaping and free regions in `value` using `f`, and
|
|
|
|
/// sets `skipped_regions` to true if any late-bound region was found
|
|
|
|
/// and skipped.
|
|
|
|
pub fn fold_regions<T,F>(&self,
|
|
|
|
value: &T,
|
|
|
|
skipped_regions: &mut bool,
|
|
|
|
mut f: F)
|
|
|
|
-> T
|
|
|
|
where F : FnMut(ty::Region, u32) -> ty::Region,
|
|
|
|
T : TypeFoldable<'tcx>,
|
|
|
|
{
|
|
|
|
value.fold_with(&mut RegionFolder::new(self, skipped_regions, &mut f))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-20 15:26:14 -05:00
|
|
|
/// Folds over the substructure of a type, visiting its component
|
|
|
|
/// types and all regions that occur *free* within it.
|
|
|
|
///
|
2014-09-13 13:09:25 -05:00
|
|
|
/// That is, `Ty` can contain function or method types that bind
|
2014-06-20 15:26:14 -05:00
|
|
|
/// regions at the call site (`ReLateBound`), and occurrences of
|
|
|
|
/// regions (aka "lifetimes") that are bound within a type are not
|
|
|
|
/// visited by this folder; only regions that occur free will be
|
|
|
|
/// visited by `fld_r`.
|
2014-12-14 06:17:23 -06:00
|
|
|
|
|
|
|
pub struct RegionFolder<'a, 'tcx: 'a> {
|
2014-04-22 07:56:37 -05:00
|
|
|
tcx: &'a ty::ctxt<'tcx>,
|
2015-06-29 16:32:39 -05:00
|
|
|
skipped_regions: &'a mut bool,
|
2014-12-04 14:06:42 -06:00
|
|
|
current_depth: u32,
|
|
|
|
fld_r: &'a mut (FnMut(ty::Region, u32) -> ty::Region + 'a),
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
2014-12-14 06:17:23 -06:00
|
|
|
impl<'a, 'tcx> RegionFolder<'a, 'tcx> {
|
2015-06-29 16:32:39 -05:00
|
|
|
pub fn new<F>(tcx: &'a ty::ctxt<'tcx>,
|
|
|
|
skipped_regions: &'a mut bool,
|
|
|
|
fld_r: &'a mut F) -> RegionFolder<'a, 'tcx>
|
2014-12-04 14:06:42 -06:00
|
|
|
where F : FnMut(ty::Region, u32) -> ty::Region
|
2014-12-14 06:17:23 -06:00
|
|
|
{
|
2013-10-29 04:25:18 -05:00
|
|
|
RegionFolder {
|
|
|
|
tcx: tcx,
|
2015-06-29 16:32:39 -05:00
|
|
|
skipped_regions: skipped_regions,
|
2014-11-15 15:47:59 -06:00
|
|
|
current_depth: 1,
|
2014-06-20 15:26:14 -05:00
|
|
|
fld_r: fld_r,
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
}
|
2014-06-20 15:26:14 -05:00
|
|
|
}
|
|
|
|
|
2014-12-14 06:17:23 -06:00
|
|
|
impl<'a, 'tcx> TypeFolder<'tcx> for RegionFolder<'a, 'tcx>
|
2014-12-08 19:26:43 -06:00
|
|
|
{
|
2014-12-16 16:34:47 -06:00
|
|
|
fn tcx(&self) -> &ty::ctxt<'tcx> { self.tcx }
|
2013-10-29 04:25:18 -05:00
|
|
|
|
2014-11-15 15:47:59 -06:00
|
|
|
fn enter_region_binder(&mut self) {
|
|
|
|
self.current_depth += 1;
|
|
|
|
}
|
2014-06-20 15:26:14 -05:00
|
|
|
|
2014-11-15 15:47:59 -06:00
|
|
|
fn exit_region_binder(&mut self) {
|
|
|
|
self.current_depth -= 1;
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
2014-06-20 15:26:14 -05:00
|
|
|
match r {
|
2014-11-15 15:47:59 -06:00
|
|
|
ty::ReLateBound(debruijn, _) if debruijn.depth < self.current_depth => {
|
2015-06-18 12:25:05 -05:00
|
|
|
debug!("RegionFolder.fold_region({:?}) skipped bound region (current depth={})",
|
|
|
|
r, self.current_depth);
|
2015-06-29 16:32:39 -05:00
|
|
|
*self.skipped_regions = true;
|
2014-06-20 15:26:14 -05:00
|
|
|
r
|
|
|
|
}
|
|
|
|
_ => {
|
2015-06-18 12:25:05 -05:00
|
|
|
debug!("RegionFolder.fold_region({:?}) folding free region (current_depth={})",
|
|
|
|
r, self.current_depth);
|
2015-01-06 11:24:46 -06:00
|
|
|
(self.fld_r)(r, self.current_depth)
|
2014-06-20 15:26:14 -05:00
|
|
|
}
|
|
|
|
}
|
2013-10-29 04:25:18 -05:00
|
|
|
}
|
|
|
|
}
|
2014-09-12 10:42:58 -05:00
|
|
|
|
2015-06-05 18:06:14 -05:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Late-bound region replacer
|
|
|
|
|
|
|
|
// Replaces the escaping regions in a type.
|
|
|
|
|
|
|
|
struct RegionReplacer<'a, 'tcx: 'a> {
|
|
|
|
tcx: &'a ty::ctxt<'tcx>,
|
|
|
|
current_depth: u32,
|
|
|
|
fld_r: &'a mut (FnMut(ty::BoundRegion) -> ty::Region + 'a),
|
|
|
|
map: FnvHashMap<ty::BoundRegion, ty::Region>
|
|
|
|
}
|
|
|
|
|
2015-09-06 13:51:58 -05:00
|
|
|
impl<'tcx> ty::ctxt<'tcx> {
|
|
|
|
pub fn replace_late_bound_regions<T,F>(&self,
|
|
|
|
value: &Binder<T>,
|
|
|
|
mut f: F)
|
|
|
|
-> (T, FnvHashMap<ty::BoundRegion, ty::Region>)
|
|
|
|
where F : FnMut(ty::BoundRegion) -> ty::Region,
|
|
|
|
T : TypeFoldable<'tcx>,
|
|
|
|
{
|
|
|
|
debug!("replace_late_bound_regions({:?})", value);
|
|
|
|
let mut replacer = RegionReplacer::new(self, &mut f);
|
|
|
|
let result = value.skip_binder().fold_with(&mut replacer);
|
|
|
|
(result, replacer.map)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Replace any late-bound regions bound in `value` with free variants attached to scope-id
|
|
|
|
/// `scope_id`.
|
|
|
|
pub fn liberate_late_bound_regions<T>(&self,
|
|
|
|
all_outlive_scope: region::CodeExtent,
|
|
|
|
value: &Binder<T>)
|
|
|
|
-> T
|
|
|
|
where T : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
self.replace_late_bound_regions(value, |br| {
|
|
|
|
ty::ReFree(ty::FreeRegion{scope: all_outlive_scope, bound_region: br})
|
|
|
|
}).0
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Flattens two binding levels into one. So `for<'a> for<'b> Foo`
|
|
|
|
/// becomes `for<'a,'b> Foo`.
|
|
|
|
pub fn flatten_late_bound_regions<T>(&self, bound2_value: &Binder<Binder<T>>)
|
|
|
|
-> Binder<T>
|
|
|
|
where T: TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
let bound0_value = bound2_value.skip_binder().skip_binder();
|
|
|
|
let value = self.fold_regions(bound0_value, &mut false,
|
|
|
|
|region, current_depth| {
|
|
|
|
match region {
|
|
|
|
ty::ReLateBound(debruijn, br) if debruijn.depth >= current_depth => {
|
|
|
|
// should be true if no escaping regions from bound2_value
|
|
|
|
assert!(debruijn.depth - current_depth <= 1);
|
|
|
|
ty::ReLateBound(ty::DebruijnIndex::new(current_depth), br)
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
region
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
Binder(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn no_late_bound_regions<T>(&self, value: &Binder<T>) -> Option<T>
|
|
|
|
where T : TypeFoldable<'tcx> + RegionEscape
|
|
|
|
{
|
|
|
|
if value.0.has_escaping_regions() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(value.0.clone())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Replace any late-bound regions bound in `value` with `'static`. Useful in trans but also
|
|
|
|
/// method lookup and a few other places where precise region relationships are not required.
|
|
|
|
pub fn erase_late_bound_regions<T>(&self, value: &Binder<T>) -> T
|
|
|
|
where T : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
self.replace_late_bound_regions(value, |_| ty::ReStatic).0
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Rewrite any late-bound regions so that they are anonymous. Region numbers are
|
|
|
|
/// assigned starting at 1 and increasing monotonically in the order traversed
|
|
|
|
/// by the fold operation.
|
|
|
|
///
|
|
|
|
/// The chief purpose of this function is to canonicalize regions so that two
|
|
|
|
/// `FnSig`s or `TraitRef`s which are equivalent up to region naming will become
|
|
|
|
/// structurally identical. For example, `for<'a, 'b> fn(&'a isize, &'b isize)` and
|
|
|
|
/// `for<'a, 'b> fn(&'b isize, &'a isize)` will become identical after anonymization.
|
|
|
|
pub fn anonymize_late_bound_regions<T>(&self, sig: &Binder<T>) -> Binder<T>
|
|
|
|
where T : TypeFoldable<'tcx>,
|
|
|
|
{
|
|
|
|
let mut counter = 0;
|
|
|
|
Binder(self.replace_late_bound_regions(sig, |_| {
|
|
|
|
counter += 1;
|
|
|
|
ty::ReLateBound(ty::DebruijnIndex::new(1), ty::BrAnon(counter))
|
|
|
|
}).0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-05 18:06:14 -05:00
|
|
|
impl<'a, 'tcx> RegionReplacer<'a, 'tcx> {
|
|
|
|
fn new<F>(tcx: &'a ty::ctxt<'tcx>, fld_r: &'a mut F) -> RegionReplacer<'a, 'tcx>
|
|
|
|
where F : FnMut(ty::BoundRegion) -> ty::Region
|
|
|
|
{
|
|
|
|
RegionReplacer {
|
|
|
|
tcx: tcx,
|
|
|
|
current_depth: 1,
|
|
|
|
fld_r: fld_r,
|
|
|
|
map: FnvHashMap()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, 'tcx> TypeFolder<'tcx> for RegionReplacer<'a, 'tcx>
|
|
|
|
{
|
|
|
|
fn tcx(&self) -> &ty::ctxt<'tcx> { self.tcx }
|
|
|
|
|
|
|
|
fn enter_region_binder(&mut self) {
|
|
|
|
self.current_depth += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn exit_region_binder(&mut self) {
|
|
|
|
self.current_depth -= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_ty(&mut self, t: Ty<'tcx>) -> Ty<'tcx> {
|
2015-06-23 18:54:32 -05:00
|
|
|
if !t.has_regions_escaping_depth(self.current_depth-1) {
|
2015-06-05 18:06:14 -05:00
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
super_fold_ty(self, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
|
|
|
match r {
|
|
|
|
ty::ReLateBound(debruijn, br) if debruijn.depth == self.current_depth => {
|
2015-06-18 12:25:05 -05:00
|
|
|
debug!("RegionReplacer.fold_region({:?}) folding region (current_depth={})",
|
|
|
|
r, self.current_depth);
|
2015-06-05 18:06:14 -05:00
|
|
|
let fld_r = &mut self.fld_r;
|
|
|
|
let region = *self.map.entry(br).or_insert_with(|| fld_r(br));
|
|
|
|
if let ty::ReLateBound(debruijn1, br) = region {
|
|
|
|
// If the callback returns a late-bound region,
|
|
|
|
// that region should always use depth 1. Then we
|
|
|
|
// adjust it to the correct depth.
|
|
|
|
assert_eq!(debruijn1.depth, 1);
|
|
|
|
ty::ReLateBound(debruijn, br)
|
|
|
|
} else {
|
|
|
|
region
|
|
|
|
}
|
|
|
|
}
|
|
|
|
r => r
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-12 10:42:58 -05:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Region eraser
|
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
impl<'tcx> ty::ctxt<'tcx> {
|
|
|
|
/// Returns an equivalent value with all free regions removed (note
|
|
|
|
/// that late-bound regions remain, because they are important for
|
|
|
|
/// subtyping, but they are anonymized and normalized as well)..
|
|
|
|
pub fn erase_regions<T>(&self, value: &T) -> T
|
|
|
|
where T : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
let value1 = value.fold_with(&mut RegionEraser(self));
|
|
|
|
debug!("erase_regions({:?}) = {:?}",
|
|
|
|
value, value1);
|
|
|
|
return value1;
|
2014-09-12 10:42:58 -05:00
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
struct RegionEraser<'a, 'tcx: 'a>(&'a ty::ctxt<'tcx>);
|
2014-09-12 10:42:58 -05:00
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
impl<'a, 'tcx> TypeFolder<'tcx> for RegionEraser<'a, 'tcx> {
|
|
|
|
fn tcx(&self) -> &ty::ctxt<'tcx> { self.0 }
|
2014-09-12 10:42:58 -05:00
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
fn fold_ty(&mut self, ty: Ty<'tcx>) -> Ty<'tcx> {
|
|
|
|
match self.tcx().normalized_cache.borrow().get(&ty).cloned() {
|
|
|
|
None => {}
|
|
|
|
Some(u) => return u
|
|
|
|
}
|
2015-06-05 18:06:14 -05:00
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
let t_norm = ty::fold::super_fold_ty(self, ty);
|
|
|
|
self.tcx().normalized_cache.borrow_mut().insert(ty, t_norm);
|
|
|
|
return t_norm;
|
|
|
|
}
|
2015-06-05 18:06:14 -05:00
|
|
|
|
2015-09-14 16:47:14 -05:00
|
|
|
fn fold_binder<T>(&mut self, t: &ty::Binder<T>) -> ty::Binder<T>
|
|
|
|
where T : TypeFoldable<'tcx>
|
|
|
|
{
|
|
|
|
let u = self.tcx().anonymize_late_bound_regions(t);
|
|
|
|
ty::fold::super_fold_binder(self, &u)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
|
|
|
// because late-bound regions affect subtyping, we can't
|
|
|
|
// erase the bound/free distinction, but we can replace
|
|
|
|
// all free regions with 'static.
|
|
|
|
//
|
|
|
|
// Note that we *CAN* replace early-bound regions -- the
|
|
|
|
// type system never "sees" those, they get substituted
|
|
|
|
// away. In trans, they will always be erased to 'static
|
|
|
|
// whenever a substitution occurs.
|
|
|
|
match r {
|
|
|
|
ty::ReLateBound(..) => r,
|
|
|
|
_ => ty::ReStatic
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_substs(&mut self,
|
|
|
|
substs: &subst::Substs<'tcx>)
|
|
|
|
-> subst::Substs<'tcx> {
|
|
|
|
subst::Substs { regions: subst::ErasedRegions,
|
|
|
|
types: substs.types.fold_with(self) }
|
|
|
|
}
|
2014-09-12 10:42:58 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-11-15 15:47:59 -06:00
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Region shifter
|
|
|
|
//
|
|
|
|
// Shifts the De Bruijn indices on all escaping bound regions by a
|
|
|
|
// fixed amount. Useful in substitution or when otherwise introducing
|
|
|
|
// a binding level that is not intended to capture the existing bound
|
|
|
|
// regions. See comment on `shift_regions_through_binders` method in
|
|
|
|
// `subst.rs` for more details.
|
|
|
|
|
2014-12-04 14:06:42 -06:00
|
|
|
pub fn shift_region(region: ty::Region, amount: u32) -> ty::Region {
|
2014-11-15 15:47:59 -06:00
|
|
|
match region {
|
|
|
|
ty::ReLateBound(debruijn, br) => {
|
|
|
|
ty::ReLateBound(debruijn.shifted(amount), br)
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
region
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-18 00:51:23 -05:00
|
|
|
pub fn shift_regions<'tcx, T:TypeFoldable<'tcx>>(tcx: &ty::ctxt<'tcx>,
|
|
|
|
amount: u32, value: &T) -> T {
|
2015-06-18 12:25:05 -05:00
|
|
|
debug!("shift_regions(value={:?}, amount={})",
|
|
|
|
value, amount);
|
2014-11-15 15:47:59 -06:00
|
|
|
|
2015-06-29 16:32:39 -05:00
|
|
|
value.fold_with(&mut RegionFolder::new(tcx, &mut false, &mut |region, _current_depth| {
|
2014-11-15 15:47:59 -06:00
|
|
|
shift_region(region, amount)
|
|
|
|
}))
|
|
|
|
}
|