Skip to content

Commit b8c1e40

Browse files
committed
Make type walking infrastructure a bit nicer
If/when we switch to using Chalk's Ty, we'll need to replace this by its `Fold` trait, but I didn't want to import the whole thing just yet.
1 parent c4fcfa2 commit b8c1e40

File tree

10 files changed

+134
-135
lines changed

10 files changed

+134
-135
lines changed

crates/ra_cli/src/analysis_stats.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use std::{collections::HashSet, fmt::Write, path::Path, time::Instant};
22

33
use ra_db::SourceDatabase;
4-
use ra_hir::{Crate, HasBodySource, HasSource, HirDisplay, ImplItem, ModuleDef, Ty};
4+
use ra_hir::{Crate, HasBodySource, HasSource, HirDisplay, ImplItem, ModuleDef, Ty, TypeWalk};
55
use ra_syntax::AstNode;
66

77
use crate::Result;

crates/ra_hir/src/lib.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,9 @@ pub use self::{
6969
resolve::Resolution,
7070
source_binder::{PathResolution, ScopeEntryWithSyntax, SourceAnalyzer},
7171
source_id::{AstIdMap, ErasedFileAstId},
72-
ty::{display::HirDisplay, ApplicationTy, CallableDef, Substs, TraitRef, Ty, TypeCtor},
72+
ty::{
73+
display::HirDisplay, ApplicationTy, CallableDef, Substs, TraitRef, Ty, TypeCtor, TypeWalk,
74+
},
7375
type_ref::Mutability,
7476
};
7577

crates/ra_hir/src/ty.rs

+120-113
Original file line numberDiff line numberDiff line change
@@ -130,12 +130,14 @@ impl ProjectionTy {
130130
substs: self.parameters.clone(),
131131
}
132132
}
133+
}
133134

134-
pub fn walk(&self, f: &mut impl FnMut(&Ty)) {
135+
impl TypeWalk for ProjectionTy {
136+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
135137
self.parameters.walk(f);
136138
}
137139

138-
pub fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
140+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
139141
self.parameters.walk_mut(f);
140142
}
141143
}
@@ -146,12 +148,12 @@ pub struct UnselectedProjectionTy {
146148
pub parameters: Substs,
147149
}
148150

149-
impl UnselectedProjectionTy {
150-
pub fn walk(&self, f: &mut impl FnMut(&Ty)) {
151+
impl TypeWalk for UnselectedProjectionTy {
152+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
151153
self.parameters.walk(f);
152154
}
153155

154-
pub fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
156+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
155157
self.parameters.walk_mut(f);
156158
}
157159
}
@@ -312,20 +314,14 @@ impl TraitRef {
312314
pub fn self_ty(&self) -> &Ty {
313315
&self.substs[0]
314316
}
317+
}
315318

316-
pub fn subst(mut self, substs: &Substs) -> TraitRef {
317-
self.substs.walk_mut(&mut |ty_mut| {
318-
let ty = mem::replace(ty_mut, Ty::Unknown);
319-
*ty_mut = ty.subst(substs);
320-
});
321-
self
322-
}
323-
324-
pub fn walk(&self, f: &mut impl FnMut(&Ty)) {
319+
impl TypeWalk for TraitRef {
320+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
325321
self.substs.walk(f);
326322
}
327323

328-
pub fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
324+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
329325
self.substs.walk_mut(f);
330326
}
331327
}
@@ -365,28 +361,18 @@ impl GenericPredicate {
365361
GenericPredicate::Error => None,
366362
}
367363
}
364+
}
368365

369-
pub fn subst(self, substs: &Substs) -> GenericPredicate {
370-
match self {
371-
GenericPredicate::Implemented(trait_ref) => {
372-
GenericPredicate::Implemented(trait_ref.subst(substs))
373-
}
374-
GenericPredicate::Projection(projection_predicate) => {
375-
GenericPredicate::Projection(projection_predicate.subst(substs))
376-
}
377-
GenericPredicate::Error => self,
378-
}
379-
}
380-
381-
pub fn walk(&self, f: &mut impl FnMut(&Ty)) {
366+
impl TypeWalk for GenericPredicate {
367+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
382368
match self {
383369
GenericPredicate::Implemented(trait_ref) => trait_ref.walk(f),
384370
GenericPredicate::Projection(projection_pred) => projection_pred.walk(f),
385371
GenericPredicate::Error => {}
386372
}
387373
}
388374

389-
pub fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
375+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
390376
match self {
391377
GenericPredicate::Implemented(trait_ref) => trait_ref.walk_mut(f),
392378
GenericPredicate::Projection(projection_pred) => projection_pred.walk_mut(f),
@@ -430,16 +416,16 @@ impl FnSig {
430416
pub fn ret(&self) -> &Ty {
431417
&self.params_and_return[self.params_and_return.len() - 1]
432418
}
419+
}
433420

434-
/// Applies the given substitutions to all types in this signature and
435-
/// returns the result.
436-
pub fn subst(&self, substs: &Substs) -> FnSig {
437-
let result: Vec<_> =
438-
self.params_and_return.iter().map(|ty| ty.clone().subst(substs)).collect();
439-
FnSig { params_and_return: result.into() }
421+
impl TypeWalk for FnSig {
422+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
423+
for t in self.params_and_return.iter() {
424+
t.walk(f);
425+
}
440426
}
441427

442-
pub fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
428+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
443429
// Without an Arc::make_mut_slice, we can't avoid the clone here:
444430
let mut v: Vec<_> = self.params_and_return.iter().cloned().collect();
445431
for t in &mut v {
@@ -463,64 +449,6 @@ impl Ty {
463449
Ty::apply(TypeCtor::Tuple { cardinality: 0 }, Substs::empty())
464450
}
465451

466-
pub fn walk(&self, f: &mut impl FnMut(&Ty)) {
467-
match self {
468-
Ty::Apply(a_ty) => {
469-
for t in a_ty.parameters.iter() {
470-
t.walk(f);
471-
}
472-
}
473-
Ty::Projection(p_ty) => {
474-
for t in p_ty.parameters.iter() {
475-
t.walk(f);
476-
}
477-
}
478-
Ty::UnselectedProjection(p_ty) => {
479-
for t in p_ty.parameters.iter() {
480-
t.walk(f);
481-
}
482-
}
483-
Ty::Dyn(predicates) | Ty::Opaque(predicates) => {
484-
for p in predicates.iter() {
485-
p.walk(f);
486-
}
487-
}
488-
Ty::Param { .. } | Ty::Bound(_) | Ty::Infer(_) | Ty::Unknown => {}
489-
}
490-
f(self);
491-
}
492-
493-
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
494-
match self {
495-
Ty::Apply(a_ty) => {
496-
a_ty.parameters.walk_mut(f);
497-
}
498-
Ty::Projection(p_ty) => {
499-
p_ty.parameters.walk_mut(f);
500-
}
501-
Ty::UnselectedProjection(p_ty) => {
502-
p_ty.parameters.walk_mut(f);
503-
}
504-
Ty::Dyn(predicates) | Ty::Opaque(predicates) => {
505-
let mut v: Vec<_> = predicates.iter().cloned().collect();
506-
for p in &mut v {
507-
p.walk_mut(f);
508-
}
509-
*predicates = v.into();
510-
}
511-
Ty::Param { .. } | Ty::Bound(_) | Ty::Infer(_) | Ty::Unknown => {}
512-
}
513-
f(self);
514-
}
515-
516-
fn fold(mut self, f: &mut impl FnMut(Ty) -> Ty) -> Ty {
517-
self.walk_mut(&mut |ty_mut| {
518-
let ty = mem::replace(ty_mut, Ty::Unknown);
519-
*ty_mut = f(ty);
520-
});
521-
self
522-
}
523-
524452
pub fn as_reference(&self) -> Option<(&Ty, Mutability)> {
525453
match self {
526454
Ty::Apply(ApplicationTy { ctor: TypeCtor::Ref(mutability), parameters }) => {
@@ -596,10 +524,53 @@ impl Ty {
596524
}
597525
}
598526

527+
/// Returns the type parameters of this type if it has some (i.e. is an ADT
528+
/// or function); so if `self` is `Option<u32>`, this returns the `u32`.
529+
pub fn substs(&self) -> Option<Substs> {
530+
match self {
531+
Ty::Apply(ApplicationTy { parameters, .. }) => Some(parameters.clone()),
532+
_ => None,
533+
}
534+
}
535+
536+
/// If this is an `impl Trait` or `dyn Trait`, returns that trait.
537+
pub fn inherent_trait(&self) -> Option<Trait> {
538+
match self {
539+
Ty::Dyn(predicates) | Ty::Opaque(predicates) => {
540+
predicates.iter().find_map(|pred| match pred {
541+
GenericPredicate::Implemented(tr) => Some(tr.trait_),
542+
_ => None,
543+
})
544+
}
545+
_ => None,
546+
}
547+
}
548+
}
549+
550+
/// This allows walking structures that contain types to do something with those
551+
/// types, similar to Chalk's `Fold` trait.
552+
pub trait TypeWalk {
553+
fn walk(&self, f: &mut impl FnMut(&Ty));
554+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty));
555+
556+
fn fold(mut self, f: &mut impl FnMut(Ty) -> Ty) -> Self
557+
where
558+
Self: Sized,
559+
{
560+
self.walk_mut(&mut |ty_mut| {
561+
let ty = mem::replace(ty_mut, Ty::Unknown);
562+
*ty_mut = f(ty);
563+
});
564+
self
565+
}
566+
599567
/// Replaces type parameters in this type using the given `Substs`. (So e.g.
600568
/// if `self` is `&[T]`, where type parameter T has index 0, and the
601569
/// `Substs` contain `u32` at index 0, we'll have `&[u32]` afterwards.)
602-
pub fn subst(self, substs: &Substs) -> Ty {
570+
fn subst(self, substs: &Substs) -> Self
571+
where
572+
Self: Sized,
573+
{
603574
self.fold(&mut |ty| match ty {
604575
Ty::Param { idx, name } => {
605576
substs.get(idx as usize).cloned().unwrap_or(Ty::Param { idx, name })
@@ -609,24 +580,21 @@ impl Ty {
609580
}
610581

611582
/// Substitutes `Ty::Bound` vars (as opposed to type parameters).
612-
pub fn subst_bound_vars(self, substs: &Substs) -> Ty {
583+
fn subst_bound_vars(self, substs: &Substs) -> Self
584+
where
585+
Self: Sized,
586+
{
613587
self.fold(&mut |ty| match ty {
614588
Ty::Bound(idx) => substs.get(idx as usize).cloned().unwrap_or_else(|| Ty::Bound(idx)),
615589
ty => ty,
616590
})
617591
}
618592

619-
/// Returns the type parameters of this type if it has some (i.e. is an ADT
620-
/// or function); so if `self` is `Option<u32>`, this returns the `u32`.
621-
pub fn substs(&self) -> Option<Substs> {
622-
match self {
623-
Ty::Apply(ApplicationTy { parameters, .. }) => Some(parameters.clone()),
624-
_ => None,
625-
}
626-
}
627-
628593
/// Shifts up `Ty::Bound` vars by `n`.
629-
pub fn shift_bound_vars(self, n: i32) -> Ty {
594+
fn shift_bound_vars(self, n: i32) -> Self
595+
where
596+
Self: Sized,
597+
{
630598
self.fold(&mut |ty| match ty {
631599
Ty::Bound(idx) => {
632600
assert!(idx as i32 >= -n);
@@ -635,18 +603,57 @@ impl Ty {
635603
ty => ty,
636604
})
637605
}
606+
}
638607

639-
/// If this is an `impl Trait` or `dyn Trait`, returns that trait.
640-
pub fn inherent_trait(&self) -> Option<Trait> {
608+
impl TypeWalk for Ty {
609+
fn walk(&self, f: &mut impl FnMut(&Ty)) {
641610
match self {
611+
Ty::Apply(a_ty) => {
612+
for t in a_ty.parameters.iter() {
613+
t.walk(f);
614+
}
615+
}
616+
Ty::Projection(p_ty) => {
617+
for t in p_ty.parameters.iter() {
618+
t.walk(f);
619+
}
620+
}
621+
Ty::UnselectedProjection(p_ty) => {
622+
for t in p_ty.parameters.iter() {
623+
t.walk(f);
624+
}
625+
}
642626
Ty::Dyn(predicates) | Ty::Opaque(predicates) => {
643-
predicates.iter().find_map(|pred| match pred {
644-
GenericPredicate::Implemented(tr) => Some(tr.trait_),
645-
_ => None,
646-
})
627+
for p in predicates.iter() {
628+
p.walk(f);
629+
}
647630
}
648-
_ => None,
631+
Ty::Param { .. } | Ty::Bound(_) | Ty::Infer(_) | Ty::Unknown => {}
632+
}
633+
f(self);
634+
}
635+
636+
fn walk_mut(&mut self, f: &mut impl FnMut(&mut Ty)) {
637+
match self {
638+
Ty::Apply(a_ty) => {
639+
a_ty.parameters.walk_mut(f);
640+
}
641+
Ty::Projection(p_ty) => {
642+
p_ty.parameters.walk_mut(f);
643+
}
644+
Ty::UnselectedProjection(p_ty) => {
645+
p_ty.parameters.walk_mut(f);
646+
}
647+
Ty::Dyn(predicates) | Ty::Opaque(predicates) => {
648+
let mut v: Vec<_> = predicates.iter().cloned().collect();
649+
for p in &mut v {
650+
p.walk_mut(f);
651+
}
652+
*predicates = v.into();
653+
}
654+
Ty::Param { .. } | Ty::Bound(_) | Ty::Infer(_) | Ty::Unknown => {}
649655
}
656+
f(self);
650657
}
651658
}
652659

crates/ra_hir/src/ty/autoderef.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ use std::iter::successors;
77

88
use log::{info, warn};
99

10-
use super::{traits::Solution, Canonical, Ty};
10+
use super::{traits::Solution, Canonical, Ty, TypeWalk};
1111
use crate::{HasGenericParams, HirDatabase, Name, Resolver};
1212

1313
const AUTODEREF_RECURSION_LIMIT: usize = 10;

crates/ra_hir/src/ty/infer.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ use super::{
3030
autoderef, lower, method_resolution, op, primitive,
3131
traits::{Guidance, Obligation, ProjectionPredicate, Solution},
3232
ApplicationTy, CallableDef, InEnvironment, ProjectionTy, Substs, TraitEnvironment, TraitRef,
33-
Ty, TypableDef, TypeCtor,
33+
Ty, TypableDef, TypeCtor, TypeWalk,
3434
};
3535
use crate::{
3636
adt::VariantDef,

crates/ra_hir/src/ty/infer/unify.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
use super::{InferenceContext, Obligation};
44
use crate::db::HirDatabase;
55
use crate::ty::{
6-
Canonical, InEnvironment, InferTy, ProjectionPredicate, ProjectionTy, TraitRef, Ty,
6+
Canonical, InEnvironment, InferTy, ProjectionPredicate, ProjectionTy, TraitRef, Ty, TypeWalk,
77
};
88

99
impl<'a, D: HirDatabase> InferenceContext<'a, D> {

crates/ra_hir/src/ty/lower.rs

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ use std::sync::Arc;
1010

1111
use super::{
1212
FnSig, GenericPredicate, ProjectionPredicate, ProjectionTy, Substs, TraitRef, Ty, TypeCtor,
13+
TypeWalk,
1314
};
1415
use crate::{
1516
adt::VariantDef,

0 commit comments

Comments
 (0)