2013-10-29 05:25:18 -04: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-09-12 10:53:35 -04: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
|
|
|
|
* behavior cannot be overriden, but it's much less work to implement.
|
|
|
|
* 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 05:25:18 -04:00
|
|
|
|
2014-05-13 11:35:42 -04:00
|
|
|
use middle::subst;
|
2014-05-31 18:53:13 -04:00
|
|
|
use middle::subst::VecPerParamSpace;
|
2013-10-29 05:25:18 -04:00
|
|
|
use middle::ty;
|
2014-09-12 10:53:35 -04:00
|
|
|
use middle::traits;
|
2014-05-12 17:12:51 -04:00
|
|
|
use middle::typeck;
|
|
|
|
use std::rc::Rc;
|
2014-06-20 22:26:14 +02:00
|
|
|
use syntax::ast;
|
2014-05-12 17:12:51 -04:00
|
|
|
use syntax::owned_slice::OwnedSlice;
|
2013-10-29 05:25:18 -04:00
|
|
|
use util::ppaux::Repr;
|
|
|
|
|
2014-05-12 17:12:51 -04: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.
|
|
|
|
pub trait TypeFoldable {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> Self;
|
2014-05-12 17:12:51 -04: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-04-22 15:56:37 +03:00
|
|
|
pub trait TypeFolder<'tcx> {
|
|
|
|
fn tcx<'a>(&'a self) -> &'a ty::ctxt<'tcx>;
|
2013-10-29 05:25:18 -04:00
|
|
|
|
|
|
|
fn fold_ty(&mut self, t: ty::t) -> ty::t {
|
|
|
|
super_fold_ty(self, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_mt(&mut self, t: &ty::mt) -> ty::mt {
|
|
|
|
super_fold_mt(self, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_trait_ref(&mut self, t: &ty::TraitRef) -> ty::TraitRef {
|
|
|
|
super_fold_trait_ref(self, t)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_sty(&mut self, sty: &ty::sty) -> ty::sty {
|
|
|
|
super_fold_sty(self, sty)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_substs(&mut self,
|
2014-05-13 11:35:42 -04:00
|
|
|
substs: &subst::Substs)
|
|
|
|
-> subst::Substs {
|
2013-10-29 05:25:18 -04:00
|
|
|
super_fold_substs(self, substs)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_sig(&mut self,
|
|
|
|
sig: &ty::FnSig)
|
|
|
|
-> ty::FnSig {
|
|
|
|
super_fold_sig(self, sig)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_bare_fn_ty(&mut self,
|
|
|
|
fty: &ty::BareFnTy)
|
2014-05-12 17:12:51 -04:00
|
|
|
-> ty::BareFnTy
|
|
|
|
{
|
|
|
|
super_fold_bare_fn_ty(self, fty)
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_closure_ty(&mut self,
|
|
|
|
fty: &ty::ClosureTy)
|
|
|
|
-> ty::ClosureTy {
|
2014-05-12 17:12:51 -04:00
|
|
|
super_fold_closure_ty(self, fty)
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
|
|
|
r
|
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_trait_store(&mut self, s: ty::TraitStore) -> ty::TraitStore {
|
|
|
|
super_fold_trait_store(self, s)
|
|
|
|
}
|
2014-05-06 15:16:11 -04:00
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
fn fold_existential_bounds(&mut self, s: ty::ExistentialBounds)
|
|
|
|
-> ty::ExistentialBounds {
|
|
|
|
super_fold_existential_bounds(self, s)
|
|
|
|
}
|
|
|
|
|
2014-05-06 15:16:11 -04:00
|
|
|
fn fold_autoref(&mut self, ar: &ty::AutoRef) -> ty::AutoRef {
|
|
|
|
super_fold_autoref(self, ar)
|
|
|
|
}
|
2014-05-12 17:12:51 -04:00
|
|
|
|
|
|
|
fn fold_item_substs(&mut self, i: ty::ItemSubsts) -> ty::ItemSubsts {
|
|
|
|
super_fold_item_substs(self, i)
|
|
|
|
}
|
2014-09-12 10:53:35 -04:00
|
|
|
|
|
|
|
fn fold_obligation(&mut self, o: &traits::Obligation) -> traits::Obligation {
|
|
|
|
super_fold_obligation(self, o)
|
|
|
|
}
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// TypeFoldable implementations.
|
|
|
|
//
|
|
|
|
// Ideally, each type should invoke `folder.fold_foo(self)` and
|
|
|
|
// nothing else. In some cases, though, we haven't gotten around to
|
|
|
|
// adding methods on the `folder` yet, and thus the folding is
|
|
|
|
// hard-coded here. This is less-flexible, because folders cannot
|
|
|
|
// override the behavior, but there are a lot of random types and one
|
|
|
|
// can easily refactor the folding into the TypeFolder trait as
|
|
|
|
// needed.
|
|
|
|
|
2014-09-12 10:53:35 -04:00
|
|
|
impl TypeFoldable for () {
|
|
|
|
fn fold_with<'tcx, F:TypeFolder<'tcx>>(&self, _: &mut F) -> () {
|
|
|
|
()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
impl<T:TypeFoldable> TypeFoldable for Option<T> {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> Option<T> {
|
2014-05-12 17:12:51 -04:00
|
|
|
self.as_ref().map(|t| t.fold_with(folder))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T:TypeFoldable> TypeFoldable for Rc<T> {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> Rc<T> {
|
2014-05-12 17:12:51 -04:00
|
|
|
Rc::new((**self).fold_with(folder))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T:TypeFoldable> TypeFoldable for Vec<T> {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> Vec<T> {
|
2014-05-12 17:12:51 -04:00
|
|
|
self.iter().map(|t| t.fold_with(folder)).collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T:TypeFoldable> TypeFoldable for OwnedSlice<T> {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> OwnedSlice<T> {
|
2014-05-12 17:12:51 -04:00
|
|
|
self.iter().map(|t| t.fold_with(folder)).collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-31 18:53:13 -04:00
|
|
|
impl<T:TypeFoldable> TypeFoldable for VecPerParamSpace<T> {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> VecPerParamSpace<T> {
|
2014-05-31 18:53:13 -04:00
|
|
|
self.map(|t| t.fold_with(folder))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
impl TypeFoldable for ty::TraitStore {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::TraitStore {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_trait_store(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::t {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::t {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_ty(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::BareFnTy {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::BareFnTy {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_bare_fn_ty(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::ClosureTy {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::ClosureTy {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_closure_ty(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::mt {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::mt {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_mt(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::FnSig {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::FnSig {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_sig(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::sty {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::sty {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_sty(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::TraitRef {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::TraitRef {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_trait_ref(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::Region {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::Region {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_region(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-13 11:35:42 -04:00
|
|
|
impl TypeFoldable for subst::Substs {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> subst::Substs {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_substs(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::ItemSubsts {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::ItemSubsts {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ItemSubsts {
|
|
|
|
substs: self.substs.fold_with(folder),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::AutoRef {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::AutoRef {
|
2014-05-12 17:12:51 -04:00
|
|
|
folder.fold_autoref(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for typeck::vtable_origin {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> typeck::vtable_origin {
|
2014-05-12 17:12:51 -04:00
|
|
|
match *self {
|
|
|
|
typeck::vtable_static(def_id, ref substs, ref origins) => {
|
|
|
|
let r_substs = substs.fold_with(folder);
|
|
|
|
let r_origins = origins.fold_with(folder);
|
|
|
|
typeck::vtable_static(def_id, r_substs, r_origins)
|
|
|
|
}
|
|
|
|
typeck::vtable_param(n, b) => {
|
|
|
|
typeck::vtable_param(n, b)
|
|
|
|
}
|
2014-05-28 22:26:56 -07:00
|
|
|
typeck::vtable_unboxed_closure(def_id) => {
|
|
|
|
typeck::vtable_unboxed_closure(def_id)
|
|
|
|
}
|
2014-05-31 18:53:13 -04:00
|
|
|
typeck::vtable_error => {
|
|
|
|
typeck::vtable_error
|
|
|
|
}
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
}
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
impl TypeFoldable for ty::BuiltinBounds {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, _folder: &mut F) -> ty::BuiltinBounds {
|
2014-05-12 17:12:51 -04:00
|
|
|
*self
|
|
|
|
}
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
impl TypeFoldable for ty::ExistentialBounds {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::ExistentialBounds {
|
2014-08-27 21:46:52 -04:00
|
|
|
folder.fold_existential_bounds(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
impl TypeFoldable for ty::ParamBounds {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::ParamBounds {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ParamBounds {
|
2014-08-27 21:46:52 -04:00
|
|
|
opt_region_bound: self.opt_region_bound.fold_with(folder),
|
2014-05-12 17:12:51 -04:00
|
|
|
builtin_bounds: self.builtin_bounds.fold_with(folder),
|
|
|
|
trait_bounds: self.trait_bounds.fold_with(folder),
|
|
|
|
}
|
|
|
|
}
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
impl TypeFoldable for ty::TypeParameterDef {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::TypeParameterDef {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::TypeParameterDef {
|
|
|
|
ident: self.ident,
|
|
|
|
def_id: self.def_id,
|
2014-05-31 18:53:13 -04:00
|
|
|
space: self.space,
|
|
|
|
index: self.index,
|
2014-05-12 17:12:51 -04:00
|
|
|
bounds: self.bounds.fold_with(folder),
|
|
|
|
default: self.default.fold_with(folder),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::RegionParameterDef {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::RegionParameterDef {
|
2014-08-27 21:46:52 -04:00
|
|
|
ty::RegionParameterDef {
|
|
|
|
name: self.name,
|
|
|
|
def_id: self.def_id,
|
|
|
|
space: self.space,
|
|
|
|
index: self.index,
|
|
|
|
bounds: self.bounds.fold_with(folder)
|
|
|
|
}
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for ty::Generics {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::Generics {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::Generics {
|
2014-05-31 18:53:13 -04:00
|
|
|
types: self.types.fold_with(folder),
|
|
|
|
regions: self.regions.fold_with(folder),
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-06 11:59:40 +02:00
|
|
|
impl TypeFoldable for ty::UnsizeKind {
|
2014-04-22 15:56:37 +03:00
|
|
|
fn fold_with<'tcx, F: TypeFolder<'tcx>>(&self, folder: &mut F) -> ty::UnsizeKind {
|
2014-08-06 11:59:40 +02:00
|
|
|
match *self {
|
|
|
|
ty::UnsizeLength(len) => ty::UnsizeLength(len),
|
|
|
|
ty::UnsizeStruct(box ref k, n) => ty::UnsizeStruct(box k.fold_with(folder), n),
|
2014-09-12 10:53:35 -04:00
|
|
|
ty::UnsizeVtable(ty::TyTrait{bounds, def_id, substs: ref substs}, self_ty) => {
|
|
|
|
ty::UnsizeVtable(
|
|
|
|
ty::TyTrait {
|
|
|
|
bounds: bounds.fold_with(folder),
|
|
|
|
def_id: def_id,
|
|
|
|
substs: substs.fold_with(folder)
|
|
|
|
},
|
|
|
|
self_ty.fold_with(folder))
|
2014-08-06 11:59:40 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-12 10:53:35 -04:00
|
|
|
impl TypeFoldable for traits::Obligation {
|
|
|
|
fn fold_with<'tcx, F:TypeFolder<'tcx>>(&self, folder: &mut F) -> traits::Obligation {
|
|
|
|
folder.fold_obligation(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N:TypeFoldable> TypeFoldable for traits::VtableImpl<N> {
|
|
|
|
fn fold_with<'tcx, F:TypeFolder<'tcx>>(&self, folder: &mut F) -> traits::VtableImpl<N> {
|
|
|
|
traits::VtableImpl {
|
|
|
|
impl_def_id: self.impl_def_id,
|
|
|
|
substs: self.substs.fold_with(folder),
|
|
|
|
nested: self.nested.fold_with(folder),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N:TypeFoldable> TypeFoldable for traits::Vtable<N> {
|
|
|
|
fn fold_with<'tcx, F:TypeFolder<'tcx>>(&self, folder: &mut F) -> traits::Vtable<N> {
|
|
|
|
match *self {
|
|
|
|
traits::VtableImpl(ref v) => traits::VtableImpl(v.fold_with(folder)),
|
|
|
|
traits::VtableUnboxedClosure(d) => traits::VtableUnboxedClosure(d),
|
|
|
|
traits::VtableParam(ref p) => traits::VtableParam(p.fold_with(folder)),
|
|
|
|
traits::VtableBuiltin => traits::VtableBuiltin,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TypeFoldable for traits::VtableParam {
|
|
|
|
fn fold_with<'tcx, F:TypeFolder<'tcx>>(&self, folder: &mut F) -> traits::VtableParam {
|
|
|
|
traits::VtableParam {
|
|
|
|
bound: self.bound.fold_with(folder),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-12 17:12:51 -04:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// "super" routines: these are the default implementations for TypeFolder.
|
|
|
|
//
|
|
|
|
// They should invoke `foo.fold_with()` to do recursive folding.
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
t: ty::t)
|
|
|
|
-> ty::t {
|
2014-05-12 17:12:51 -04:00
|
|
|
let sty = ty::get(t).sty.fold_with(this);
|
2014-03-06 05:07:47 +02:00
|
|
|
ty::mk_t(this.tcx(), sty)
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_substs<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
substs: &subst::Substs)
|
|
|
|
-> subst::Substs {
|
2013-10-29 05:25:18 -04:00
|
|
|
let regions = match substs.regions {
|
2014-05-13 11:35:42 -04:00
|
|
|
subst::ErasedRegions => {
|
|
|
|
subst::ErasedRegions
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
2014-05-13 11:35:42 -04:00
|
|
|
subst::NonerasedRegions(ref regions) => {
|
|
|
|
subst::NonerasedRegions(regions.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-05-13 11:35:42 -04:00
|
|
|
subst::Substs { regions: regions,
|
2014-05-31 18:53:13 -04:00
|
|
|
types: substs.types.fold_with(this) }
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_sig<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
sig: &ty::FnSig)
|
|
|
|
-> ty::FnSig {
|
2013-10-29 05:25:18 -04:00
|
|
|
ty::FnSig { binder_id: sig.binder_id,
|
2014-05-12 17:12:51 -04:00
|
|
|
inputs: sig.inputs.fold_with(this),
|
|
|
|
output: sig.output.fold_with(this),
|
2013-10-29 05:25:18 -04:00
|
|
|
variadic: sig.variadic }
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_bare_fn_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
fty: &ty::BareFnTy)
|
|
|
|
-> ty::BareFnTy
|
2014-05-12 17:12:51 -04:00
|
|
|
{
|
|
|
|
ty::BareFnTy { sig: fty.sig.fold_with(this),
|
|
|
|
abi: fty.abi,
|
|
|
|
fn_style: fty.fn_style }
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_closure_ty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
fty: &ty::ClosureTy)
|
|
|
|
-> ty::ClosureTy
|
2014-05-12 17:12:51 -04:00
|
|
|
{
|
|
|
|
ty::ClosureTy {
|
|
|
|
store: fty.store.fold_with(this),
|
|
|
|
sig: fty.sig.fold_with(this),
|
|
|
|
fn_style: fty.fn_style,
|
|
|
|
onceness: fty.onceness,
|
2014-08-27 21:46:52 -04:00
|
|
|
bounds: fty.bounds.fold_with(this),
|
2014-05-28 22:26:56 -07:00
|
|
|
abi: fty.abi,
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
}
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_trait_ref<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
t: &ty::TraitRef)
|
|
|
|
-> ty::TraitRef {
|
2013-10-29 05:25:18 -04:00
|
|
|
ty::TraitRef {
|
|
|
|
def_id: t.def_id,
|
2014-05-12 17:12:51 -04:00
|
|
|
substs: t.substs.fold_with(this),
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_mt<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
mt: &ty::mt) -> ty::mt {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::mt {ty: mt.ty.fold_with(this),
|
2013-10-29 05:25:18 -04:00
|
|
|
mutbl: mt.mutbl}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_sty<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
sty: &ty::sty) -> ty::sty {
|
2013-10-29 05:25:18 -04:00
|
|
|
match *sty {
|
2013-12-30 18:57:48 -08:00
|
|
|
ty::ty_box(typ) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_box(typ.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
2014-01-12 02:25:51 +02:00
|
|
|
ty::ty_uniq(typ) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_uniq(typ.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_ptr(ref tm) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_ptr(tm.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
DST coercions and DST structs
[breaking-change]
1. The internal layout for traits has changed from (vtable, data) to (data, vtable). If you were relying on this in unsafe transmutes, you might get some very weird and apparently unrelated errors. You should not be doing this! Prefer not to do this at all, but if you must, you should use raw::TraitObject rather than hardcoding rustc's internal representation into your code.
2. The minimal type of reference-to-vec-literals (e.g., `&[1, 2, 3]`) is now a fixed size vec (e.g., `&[int, ..3]`) where it used to be an unsized vec (e.g., `&[int]`). If you want the unszied type, you must explicitly give the type (e.g., `let x: &[_] = &[1, 2, 3]`). Note in particular where multiple blocks must have the same type (e.g., if and else clauses, vec elements), the compiler will not coerce to the unsized type without a hint. E.g., `[&[1], &[1, 2]]` used to be a valid expression of type '[&[int]]'. It no longer type checks since the first element now has type `&[int, ..1]` and the second has type &[int, ..2]` which are incompatible.
3. The type of blocks (including functions) must be coercible to the expected type (used to be a subtype). Mostly this makes things more flexible and not less (in particular, in the case of coercing function bodies to the return type). However, in some rare cases, this is less flexible. TBH, I'm not exactly sure of the exact effects. I think the change causes us to resolve inferred type variables slightly earlier which might make us slightly more restrictive. Possibly it only affects blocks with unreachable code. E.g., `if ... { fail!(); "Hello" }` used to type check, it no longer does. The fix is to add a semicolon after the string.
2014-08-04 14:20:11 +02:00
|
|
|
ty::ty_vec(typ, sz) => {
|
|
|
|
ty::ty_vec(typ.fold_with(this), sz)
|
|
|
|
}
|
|
|
|
ty::ty_open(typ) => {
|
|
|
|
ty::ty_open(typ.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_enum(tid, ref substs) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_enum(tid, substs.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
2014-05-05 18:56:44 -07:00
|
|
|
ty::ty_trait(box ty::TyTrait {
|
|
|
|
def_id,
|
|
|
|
ref substs,
|
|
|
|
bounds
|
|
|
|
}) => {
|
|
|
|
ty::ty_trait(box ty::TyTrait {
|
2014-03-19 22:01:30 +11:00
|
|
|
def_id: def_id,
|
2014-05-12 17:12:51 -04:00
|
|
|
substs: substs.fold_with(this),
|
2014-08-27 21:46:52 -04:00
|
|
|
bounds: this.fold_existential_bounds(bounds),
|
2014-03-19 22:01:30 +11:00
|
|
|
})
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_tup(ref ts) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_tup(ts.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_bare_fn(ref f) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_bare_fn(f.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_closure(ref f) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_closure(box f.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_rptr(r, ref tm) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_rptr(r.fold_with(this), tm.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
ty::ty_struct(did, ref substs) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::ty_struct(did, substs.fold_with(this))
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
2014-07-29 22:08:39 -07:00
|
|
|
ty::ty_unboxed_closure(did, ref region) => {
|
|
|
|
ty::ty_unboxed_closure(did, region.fold_with(this))
|
2014-05-28 22:26:56 -07:00
|
|
|
}
|
2014-04-29 13:10:23 +12:00
|
|
|
ty::ty_nil | ty::ty_bot | ty::ty_bool | ty::ty_char | ty::ty_str |
|
2014-02-11 13:48:55 +02:00
|
|
|
ty::ty_int(_) | ty::ty_uint(_) | ty::ty_float(_) |
|
2014-01-11 16:39:32 +02:00
|
|
|
ty::ty_err | ty::ty_infer(_) |
|
2014-05-31 18:53:13 -04:00
|
|
|
ty::ty_param(..) => {
|
2013-10-29 05:25:18 -04:00
|
|
|
(*sty).clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_trait_store<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
trait_store: ty::TraitStore)
|
|
|
|
-> ty::TraitStore {
|
2013-10-29 05:25:18 -04:00
|
|
|
match trait_store {
|
2014-04-11 09:01:31 +03:00
|
|
|
ty::UniqTraitStore => ty::UniqTraitStore,
|
|
|
|
ty::RegionTraitStore(r, m) => {
|
2014-05-12 17:12:51 -04:00
|
|
|
ty::RegionTraitStore(r.fold_with(this), m)
|
2014-04-11 09:01:31 +03:00
|
|
|
}
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_existential_bounds<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
bounds: ty::ExistentialBounds)
|
|
|
|
-> ty::ExistentialBounds {
|
2014-08-27 21:46:52 -04:00
|
|
|
ty::ExistentialBounds {
|
|
|
|
region_bound: bounds.region_bound.fold_with(this),
|
|
|
|
builtin_bounds: bounds.builtin_bounds,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_autoref<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
autoref: &ty::AutoRef)
|
|
|
|
-> ty::AutoRef
|
2014-05-06 15:16:11 -04:00
|
|
|
{
|
|
|
|
match *autoref {
|
DST coercions and DST structs
[breaking-change]
1. The internal layout for traits has changed from (vtable, data) to (data, vtable). If you were relying on this in unsafe transmutes, you might get some very weird and apparently unrelated errors. You should not be doing this! Prefer not to do this at all, but if you must, you should use raw::TraitObject rather than hardcoding rustc's internal representation into your code.
2. The minimal type of reference-to-vec-literals (e.g., `&[1, 2, 3]`) is now a fixed size vec (e.g., `&[int, ..3]`) where it used to be an unsized vec (e.g., `&[int]`). If you want the unszied type, you must explicitly give the type (e.g., `let x: &[_] = &[1, 2, 3]`). Note in particular where multiple blocks must have the same type (e.g., if and else clauses, vec elements), the compiler will not coerce to the unsized type without a hint. E.g., `[&[1], &[1, 2]]` used to be a valid expression of type '[&[int]]'. It no longer type checks since the first element now has type `&[int, ..1]` and the second has type &[int, ..2]` which are incompatible.
3. The type of blocks (including functions) must be coercible to the expected type (used to be a subtype). Mostly this makes things more flexible and not less (in particular, in the case of coercing function bodies to the return type). However, in some rare cases, this is less flexible. TBH, I'm not exactly sure of the exact effects. I think the change causes us to resolve inferred type variables slightly earlier which might make us slightly more restrictive. Possibly it only affects blocks with unreachable code. E.g., `if ... { fail!(); "Hello" }` used to type check, it no longer does. The fix is to add a semicolon after the string.
2014-08-04 14:20:11 +02:00
|
|
|
ty::AutoPtr(r, m, None) => ty::AutoPtr(this.fold_region(r), m, None),
|
|
|
|
ty::AutoPtr(r, m, Some(ref a)) => {
|
2014-08-06 11:59:40 +02:00
|
|
|
ty::AutoPtr(this.fold_region(r), m, Some(box super_fold_autoref(this, &**a)))
|
DST coercions and DST structs
[breaking-change]
1. The internal layout for traits has changed from (vtable, data) to (data, vtable). If you were relying on this in unsafe transmutes, you might get some very weird and apparently unrelated errors. You should not be doing this! Prefer not to do this at all, but if you must, you should use raw::TraitObject rather than hardcoding rustc's internal representation into your code.
2. The minimal type of reference-to-vec-literals (e.g., `&[1, 2, 3]`) is now a fixed size vec (e.g., `&[int, ..3]`) where it used to be an unsized vec (e.g., `&[int]`). If you want the unszied type, you must explicitly give the type (e.g., `let x: &[_] = &[1, 2, 3]`). Note in particular where multiple blocks must have the same type (e.g., if and else clauses, vec elements), the compiler will not coerce to the unsized type without a hint. E.g., `[&[1], &[1, 2]]` used to be a valid expression of type '[&[int]]'. It no longer type checks since the first element now has type `&[int, ..1]` and the second has type &[int, ..2]` which are incompatible.
3. The type of blocks (including functions) must be coercible to the expected type (used to be a subtype). Mostly this makes things more flexible and not less (in particular, in the case of coercing function bodies to the return type). However, in some rare cases, this is less flexible. TBH, I'm not exactly sure of the exact effects. I think the change causes us to resolve inferred type variables slightly earlier which might make us slightly more restrictive. Possibly it only affects blocks with unreachable code. E.g., `if ... { fail!(); "Hello" }` used to type check, it no longer does. The fix is to add a semicolon after the string.
2014-08-04 14:20:11 +02:00
|
|
|
}
|
2014-08-27 17:07:28 +12:00
|
|
|
ty::AutoUnsafe(m, None) => ty::AutoUnsafe(m, None),
|
|
|
|
ty::AutoUnsafe(m, Some(ref a)) => {
|
|
|
|
ty::AutoUnsafe(m, Some(box super_fold_autoref(this, &**a)))
|
|
|
|
}
|
2014-08-06 11:59:40 +02:00
|
|
|
ty::AutoUnsize(ref k) => ty::AutoUnsize(k.fold_with(this)),
|
|
|
|
ty::AutoUnsizeUniq(ref k) => ty::AutoUnsizeUniq(k.fold_with(this)),
|
2014-05-12 17:12:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn super_fold_item_substs<'tcx, T: TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
substs: ty::ItemSubsts)
|
|
|
|
-> ty::ItemSubsts
|
2014-05-12 17:12:51 -04:00
|
|
|
{
|
|
|
|
ty::ItemSubsts {
|
|
|
|
substs: substs.substs.fold_with(this),
|
2014-05-06 15:16:11 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-12 10:53:35 -04:00
|
|
|
pub fn super_fold_obligation<'tcx, T:TypeFolder<'tcx>>(this: &mut T,
|
|
|
|
obligation: &traits::Obligation)
|
|
|
|
-> traits::Obligation
|
|
|
|
{
|
|
|
|
traits::Obligation {
|
|
|
|
cause: obligation.cause,
|
|
|
|
recursion_depth: obligation.recursion_depth,
|
|
|
|
trait_ref: obligation.trait_ref.fold_with(this),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-29 05:25:18 -04:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Some sample folders
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub struct BottomUpFolder<'a, 'tcx: 'a> {
|
|
|
|
pub tcx: &'a ty::ctxt<'tcx>,
|
2014-04-07 13:30:48 -07:00
|
|
|
pub fldop: |ty::t|: 'a -> ty::t,
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
impl<'a, 'tcx> TypeFolder<'tcx> for BottomUpFolder<'a, 'tcx> {
|
|
|
|
fn tcx<'a>(&'a self) -> &'a ty::ctxt<'tcx> { self.tcx }
|
2013-10-29 05:25:18 -04:00
|
|
|
|
|
|
|
fn fold_ty(&mut self, ty: ty::t) -> ty::t {
|
|
|
|
let t1 = super_fold_ty(self, ty);
|
|
|
|
(self.fldop)(t1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// Region folder
|
|
|
|
|
2014-06-20 22:26:14 +02:00
|
|
|
/// Folds over the substructure of a type, visiting its component
|
|
|
|
/// types and all regions that occur *free* within it.
|
|
|
|
///
|
|
|
|
/// That is, `ty::t` can contain function or method types that bind
|
|
|
|
/// 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`.
|
|
|
|
///
|
|
|
|
/// (The distinction between "free" and "bound" is represented by
|
|
|
|
/// keeping track of each `FnSig` in the lexical context of the
|
|
|
|
/// current position of the fold.)
|
2014-04-22 15:56:37 +03:00
|
|
|
pub struct RegionFolder<'a, 'tcx: 'a> {
|
|
|
|
tcx: &'a ty::ctxt<'tcx>,
|
2014-04-07 13:30:48 -07:00
|
|
|
fld_t: |ty::t|: 'a -> ty::t,
|
|
|
|
fld_r: |ty::Region|: 'a -> ty::Region,
|
2014-06-20 22:26:14 +02:00
|
|
|
within_binder_ids: Vec<ast::NodeId>,
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
impl<'a, 'tcx> RegionFolder<'a, 'tcx> {
|
|
|
|
pub fn general(tcx: &'a ty::ctxt<'tcx>,
|
2014-04-07 13:30:48 -07:00
|
|
|
fld_r: |ty::Region|: 'a -> ty::Region,
|
|
|
|
fld_t: |ty::t|: 'a -> ty::t)
|
2014-04-22 15:56:37 +03:00
|
|
|
-> RegionFolder<'a, 'tcx> {
|
2013-10-29 05:25:18 -04:00
|
|
|
RegionFolder {
|
|
|
|
tcx: tcx,
|
|
|
|
fld_t: fld_t,
|
2014-06-20 22:26:14 +02:00
|
|
|
fld_r: fld_r,
|
|
|
|
within_binder_ids: vec![],
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
pub fn regions(tcx: &'a ty::ctxt<'tcx>, fld_r: |ty::Region|: 'a -> ty::Region)
|
|
|
|
-> RegionFolder<'a, 'tcx> {
|
2013-10-29 05:25:18 -04:00
|
|
|
fn noop(t: ty::t) -> ty::t { t }
|
|
|
|
|
|
|
|
RegionFolder {
|
|
|
|
tcx: tcx,
|
|
|
|
fld_t: noop,
|
2014-06-20 22:26:14 +02:00
|
|
|
fld_r: fld_r,
|
|
|
|
within_binder_ids: vec![],
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-20 22:26:14 +02:00
|
|
|
/// If `ty` has `FnSig` (i.e. closure or fn), return its binder_id;
|
|
|
|
/// else None.
|
|
|
|
fn opt_binder_id_of_function(t: ty::t) -> Option<ast::NodeId> {
|
|
|
|
match ty::get(t).sty {
|
|
|
|
ty::ty_closure(ref f) => Some(f.sig.binder_id),
|
|
|
|
ty::ty_bare_fn(ref f) => Some(f.sig.binder_id),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 15:56:37 +03:00
|
|
|
impl<'a, 'tcx> TypeFolder<'tcx> for RegionFolder<'a, 'tcx> {
|
|
|
|
fn tcx<'a>(&'a self) -> &'a ty::ctxt<'tcx> { self.tcx }
|
2013-10-29 05:25:18 -04:00
|
|
|
|
|
|
|
fn fold_ty(&mut self, ty: ty::t) -> ty::t {
|
|
|
|
debug!("RegionFolder.fold_ty({})", ty.repr(self.tcx()));
|
2014-06-20 22:26:14 +02:00
|
|
|
let opt_binder_id = opt_binder_id_of_function(ty);
|
|
|
|
match opt_binder_id {
|
|
|
|
Some(binder_id) => self.within_binder_ids.push(binder_id),
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
|
2013-10-29 05:25:18 -04:00
|
|
|
let t1 = super_fold_ty(self, ty);
|
2014-06-20 22:26:14 +02:00
|
|
|
let ret = (self.fld_t)(t1);
|
|
|
|
|
|
|
|
if opt_binder_id.is_some() {
|
|
|
|
self.within_binder_ids.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
ret
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fn fold_region(&mut self, r: ty::Region) -> ty::Region {
|
2014-06-20 22:26:14 +02:00
|
|
|
match r {
|
|
|
|
ty::ReLateBound(binder_id, _) if self.within_binder_ids.contains(&binder_id) => {
|
|
|
|
debug!("RegionFolder.fold_region({}) skipped bound region", r.repr(self.tcx()));
|
|
|
|
r
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
debug!("RegionFolder.fold_region({}) folding free region", r.repr(self.tcx()));
|
|
|
|
(self.fld_r)(r)
|
|
|
|
}
|
|
|
|
}
|
2013-10-29 05:25:18 -04:00
|
|
|
}
|
|
|
|
}
|