mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-22 06:44:35 +00:00
ty/walk: keep track of GenericArg
s on the stack, instead of Ty
s.
This commit is contained in:
parent
40158901d5
commit
e53c42c0b3
@ -2698,14 +2698,14 @@ impl<'tcx> TyS<'tcx> {
|
||||
/// [isize] => { [isize], isize }
|
||||
/// ```
|
||||
pub fn walk(&'tcx self) -> TypeWalker<'tcx> {
|
||||
TypeWalker::new(self)
|
||||
TypeWalker::new(self.into())
|
||||
}
|
||||
|
||||
/// Iterator that walks the immediate children of `self`. Hence
|
||||
/// `Foo<Bar<i32>, u32>` yields the sequence `[Bar<i32>, u32]`
|
||||
/// (but not `i32`, like `walk`).
|
||||
pub fn walk_shallow(&'tcx self) -> smallvec::IntoIter<walk::TypeWalkerArray<'tcx>> {
|
||||
walk::walk_shallow(self)
|
||||
pub fn walk_shallow(&'tcx self) -> impl Iterator<Item = Ty<'tcx>> {
|
||||
walk::walk_shallow(self.into())
|
||||
}
|
||||
|
||||
/// Walks `ty` and any types appearing within `ty`, invoking the
|
||||
|
@ -1,13 +1,13 @@
|
||||
//! An iterator over the type substructure.
|
||||
//! WARNING: this does not keep track of the region depth.
|
||||
|
||||
use crate::ty::subst::{GenericArg, GenericArgKind};
|
||||
use crate::ty::{self, Ty};
|
||||
use smallvec::{self, SmallVec};
|
||||
|
||||
// The TypeWalker's stack is hot enough that it's worth going to some effort to
|
||||
// avoid heap allocations.
|
||||
pub type TypeWalkerArray<'tcx> = [Ty<'tcx>; 8];
|
||||
pub type TypeWalkerStack<'tcx> = SmallVec<TypeWalkerArray<'tcx>>;
|
||||
type TypeWalkerStack<'tcx> = SmallVec<[GenericArg<'tcx>; 8]>;
|
||||
|
||||
pub struct TypeWalker<'tcx> {
|
||||
stack: TypeWalkerStack<'tcx>,
|
||||
@ -15,11 +15,11 @@ pub struct TypeWalker<'tcx> {
|
||||
}
|
||||
|
||||
impl<'tcx> TypeWalker<'tcx> {
|
||||
pub fn new(ty: Ty<'tcx>) -> TypeWalker<'tcx> {
|
||||
TypeWalker { stack: smallvec![ty], last_subtree: 1 }
|
||||
pub fn new(root: GenericArg<'tcx>) -> TypeWalker<'tcx> {
|
||||
TypeWalker { stack: smallvec![root], last_subtree: 1 }
|
||||
}
|
||||
|
||||
/// Skips the subtree of types corresponding to the last type
|
||||
/// Skips the subtree corresponding to the last type
|
||||
/// returned by `next()`.
|
||||
///
|
||||
/// Example: Imagine you are walking `Foo<Bar<int>, usize>`.
|
||||
@ -41,98 +41,120 @@ impl<'tcx> Iterator for TypeWalker<'tcx> {
|
||||
|
||||
fn next(&mut self) -> Option<Ty<'tcx>> {
|
||||
debug!("next(): stack={:?}", self.stack);
|
||||
match self.stack.pop() {
|
||||
None => None,
|
||||
Some(ty) => {
|
||||
self.last_subtree = self.stack.len();
|
||||
push_subtypes(&mut self.stack, ty);
|
||||
debug!("next: stack={:?}", self.stack);
|
||||
Some(ty)
|
||||
while let Some(next) = self.stack.pop() {
|
||||
self.last_subtree = self.stack.len();
|
||||
push_inner(&mut self.stack, next);
|
||||
debug!("next: stack={:?}", self.stack);
|
||||
|
||||
// FIXME(eddyb) remove this filter and expose all `GenericArg`s.
|
||||
match next.unpack() {
|
||||
GenericArgKind::Type(ty) => return Some(ty),
|
||||
GenericArgKind::Lifetime(_) | GenericArgKind::Const(_) => {}
|
||||
}
|
||||
}
|
||||
|
||||
None
|
||||
}
|
||||
}
|
||||
|
||||
pub fn walk_shallow(ty: Ty<'_>) -> smallvec::IntoIter<TypeWalkerArray<'_>> {
|
||||
pub fn walk_shallow(parent: GenericArg<'tcx>) -> impl Iterator<Item = Ty<'tcx>> {
|
||||
let mut stack = SmallVec::new();
|
||||
push_subtypes(&mut stack, ty);
|
||||
stack.into_iter()
|
||||
push_inner(&mut stack, parent);
|
||||
stack.into_iter().filter_map(|child| {
|
||||
// FIXME(eddyb) remove this filter and expose all `GenericArg`s.
|
||||
match child.unpack() {
|
||||
GenericArgKind::Type(ty) => Some(ty),
|
||||
GenericArgKind::Lifetime(_) | GenericArgKind::Const(_) => None,
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
// We push types on the stack in reverse order so as to
|
||||
// We push `GenericArg`s on the stack in reverse order so as to
|
||||
// maintain a pre-order traversal. As of the time of this
|
||||
// writing, the fact that the traversal is pre-order is not
|
||||
// known to be significant to any code, but it seems like the
|
||||
// natural order one would expect (basically, the order of the
|
||||
// types as they are written).
|
||||
fn push_subtypes<'tcx>(stack: &mut TypeWalkerStack<'tcx>, parent_ty: Ty<'tcx>) {
|
||||
match parent_ty.kind {
|
||||
ty::Bool
|
||||
| ty::Char
|
||||
| ty::Int(_)
|
||||
| ty::Uint(_)
|
||||
| ty::Float(_)
|
||||
| ty::Str
|
||||
| ty::Infer(_)
|
||||
| ty::Param(_)
|
||||
| ty::Never
|
||||
| ty::Error
|
||||
| ty::Placeholder(..)
|
||||
| ty::Bound(..)
|
||||
| ty::Foreign(..) => {}
|
||||
ty::Array(ty, len) => {
|
||||
if let ty::ConstKind::Unevaluated(_, substs, promoted) = len.val {
|
||||
assert!(promoted.is_none());
|
||||
stack.extend(substs.types().rev());
|
||||
}
|
||||
stack.push(len.ty);
|
||||
stack.push(ty);
|
||||
}
|
||||
ty::Slice(ty) => {
|
||||
stack.push(ty);
|
||||
}
|
||||
ty::RawPtr(ref mt) => {
|
||||
stack.push(mt.ty);
|
||||
}
|
||||
ty::Ref(_, ty, _) => {
|
||||
stack.push(ty);
|
||||
}
|
||||
ty::Projection(ref data) | ty::UnnormalizedProjection(ref data) => {
|
||||
stack.extend(data.substs.types().rev());
|
||||
}
|
||||
ty::Dynamic(ref obj, ..) => {
|
||||
stack.extend(obj.iter().rev().flat_map(|predicate| {
|
||||
let (substs, opt_ty) = match *predicate.skip_binder() {
|
||||
ty::ExistentialPredicate::Trait(tr) => (tr.substs, None),
|
||||
ty::ExistentialPredicate::Projection(p) => (p.substs, Some(p.ty)),
|
||||
ty::ExistentialPredicate::AutoTrait(_) =>
|
||||
// Empty iterator
|
||||
{
|
||||
(ty::InternalSubsts::empty(), None)
|
||||
}
|
||||
};
|
||||
fn push_inner<'tcx>(stack: &mut TypeWalkerStack<'tcx>, parent: GenericArg<'tcx>) {
|
||||
match parent.unpack() {
|
||||
GenericArgKind::Type(parent_ty) => match parent_ty.kind {
|
||||
ty::Bool
|
||||
| ty::Char
|
||||
| ty::Int(_)
|
||||
| ty::Uint(_)
|
||||
| ty::Float(_)
|
||||
| ty::Str
|
||||
| ty::Infer(_)
|
||||
| ty::Param(_)
|
||||
| ty::Never
|
||||
| ty::Error
|
||||
| ty::Placeholder(..)
|
||||
| ty::Bound(..)
|
||||
| ty::Foreign(..) => {}
|
||||
|
||||
substs.types().rev().chain(opt_ty)
|
||||
}));
|
||||
}
|
||||
ty::Adt(_, substs) | ty::Opaque(_, substs) => {
|
||||
stack.extend(substs.types().rev());
|
||||
}
|
||||
ty::Closure(_, ref substs) | ty::Generator(_, ref substs, _) => {
|
||||
stack.extend(substs.types().rev());
|
||||
}
|
||||
ty::GeneratorWitness(ts) => {
|
||||
stack.extend(ts.skip_binder().iter().cloned().rev());
|
||||
}
|
||||
ty::Tuple(..) => {
|
||||
stack.extend(parent_ty.tuple_fields().rev());
|
||||
}
|
||||
ty::FnDef(_, substs) => {
|
||||
stack.extend(substs.types().rev());
|
||||
}
|
||||
ty::FnPtr(sig) => {
|
||||
stack.push(sig.skip_binder().output());
|
||||
stack.extend(sig.skip_binder().inputs().iter().cloned().rev());
|
||||
ty::Array(ty, len) => {
|
||||
stack.push(len.into());
|
||||
stack.push(ty.into());
|
||||
}
|
||||
ty::Slice(ty) => {
|
||||
stack.push(ty.into());
|
||||
}
|
||||
ty::RawPtr(mt) => {
|
||||
stack.push(mt.ty.into());
|
||||
}
|
||||
ty::Ref(lt, ty, _) => {
|
||||
stack.push(ty.into());
|
||||
stack.push(lt.into());
|
||||
}
|
||||
ty::Projection(data) | ty::UnnormalizedProjection(data) => {
|
||||
stack.extend(data.substs.iter().copied().rev());
|
||||
}
|
||||
ty::Dynamic(obj, lt) => {
|
||||
stack.push(lt.into());
|
||||
stack.extend(obj.iter().rev().flat_map(|predicate| {
|
||||
let (substs, opt_ty) = match *predicate.skip_binder() {
|
||||
ty::ExistentialPredicate::Trait(tr) => (tr.substs, None),
|
||||
ty::ExistentialPredicate::Projection(p) => (p.substs, Some(p.ty)),
|
||||
ty::ExistentialPredicate::AutoTrait(_) =>
|
||||
// Empty iterator
|
||||
{
|
||||
(ty::InternalSubsts::empty(), None)
|
||||
}
|
||||
};
|
||||
|
||||
substs.iter().copied().rev().chain(opt_ty.map(|ty| ty.into()))
|
||||
}));
|
||||
}
|
||||
ty::Adt(_, substs)
|
||||
| ty::Opaque(_, substs)
|
||||
| ty::Closure(_, substs)
|
||||
| ty::Generator(_, substs, _)
|
||||
| ty::Tuple(substs)
|
||||
| ty::FnDef(_, substs) => {
|
||||
stack.extend(substs.iter().copied().rev());
|
||||
}
|
||||
ty::GeneratorWitness(ts) => {
|
||||
stack.extend(ts.skip_binder().iter().cloned().rev().map(|ty| ty.into()));
|
||||
}
|
||||
ty::FnPtr(sig) => {
|
||||
stack.push(sig.skip_binder().output().into());
|
||||
stack.extend(sig.skip_binder().inputs().iter().cloned().rev().map(|ty| ty.into()));
|
||||
}
|
||||
},
|
||||
GenericArgKind::Lifetime(_) => {}
|
||||
GenericArgKind::Const(parent_ct) => {
|
||||
stack.push(parent_ct.ty.into());
|
||||
match parent_ct.val {
|
||||
ty::ConstKind::Infer(_)
|
||||
| ty::ConstKind::Param(_)
|
||||
| ty::ConstKind::Placeholder(_)
|
||||
| ty::ConstKind::Bound(..)
|
||||
| ty::ConstKind::Value(_) => {}
|
||||
|
||||
ty::ConstKind::Unevaluated(_, substs, _) => {
|
||||
stack.extend(substs.iter().copied().rev());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user