Skip to content

Commit f7686a5

Browse files
arielb1brson
authored andcommitted
add a per-param-env cache to impls_bound
There used to be only a global cache, which led to uncached calls to trait selection when there were type parameters. I'm running a check that there are no adverse performance effects. Fixes rust-lang#37106 (drop elaboration times are now ~half of borrow checking, so might still be worthy of optimization, but not critical).
1 parent 3560363 commit f7686a5

File tree

5 files changed

+41
-27
lines changed

5 files changed

+41
-27
lines changed

src/librustc/traits/select.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -336,7 +336,7 @@ impl<'cx, 'gcx, 'tcx> SelectionContext<'cx, 'gcx, 'tcx> {
336336
self.infcx.tcx
337337
}
338338

339-
pub fn param_env(&self) -> &'cx ty::ParameterEnvironment<'tcx> {
339+
pub fn param_env(&self) -> &'cx ty::ParameterEnvironment<'gcx> {
340340
self.infcx.param_env()
341341
}
342342

src/librustc/ty/mod.rs

+14-2
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ use util::nodemap::FnvHashMap;
3434

3535
use serialize::{self, Encodable, Encoder};
3636
use std::borrow::Cow;
37-
use std::cell::Cell;
37+
use std::cell::{Cell, RefCell};
3838
use std::hash::{Hash, Hasher};
3939
use std::iter;
4040
use std::ops::Deref;
@@ -1220,6 +1220,12 @@ pub struct ParameterEnvironment<'tcx> {
12201220
/// regions don't have this implicit scope and instead introduce
12211221
/// relationships in the environment.
12221222
pub free_id_outlive: CodeExtent,
1223+
1224+
/// A cache for `moves_by_default`.
1225+
pub is_copy_cache: RefCell<FnvHashMap<Ty<'tcx>, bool>>,
1226+
1227+
/// A cache for `type_is_sized`
1228+
pub is_sized_cache: RefCell<FnvHashMap<Ty<'tcx>, bool>>,
12231229
}
12241230

12251231
impl<'a, 'tcx> ParameterEnvironment<'tcx> {
@@ -1232,6 +1238,8 @@ impl<'a, 'tcx> ParameterEnvironment<'tcx> {
12321238
implicit_region_bound: self.implicit_region_bound,
12331239
caller_bounds: caller_bounds,
12341240
free_id_outlive: self.free_id_outlive,
1241+
is_copy_cache: RefCell::new(FnvHashMap()),
1242+
is_sized_cache: RefCell::new(FnvHashMap()),
12351243
}
12361244
}
12371245

@@ -2782,7 +2790,9 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
27822790
free_substs: Substs::empty(self),
27832791
caller_bounds: Vec::new(),
27842792
implicit_region_bound: self.mk_region(ty::ReEmpty),
2785-
free_id_outlive: free_id_outlive
2793+
free_id_outlive: free_id_outlive,
2794+
is_copy_cache: RefCell::new(FnvHashMap()),
2795+
is_sized_cache: RefCell::new(FnvHashMap()),
27862796
}
27872797
}
27882798

@@ -2853,6 +2863,8 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'tcx> {
28532863
implicit_region_bound: tcx.mk_region(ty::ReScope(free_id_outlive)),
28542864
caller_bounds: predicates,
28552865
free_id_outlive: free_id_outlive,
2866+
is_copy_cache: RefCell::new(FnvHashMap()),
2867+
is_sized_cache: RefCell::new(FnvHashMap()),
28562868
};
28572869

28582870
let cause = traits::ObligationCause::misc(span, free_id_outlive.node_id(&self.region_maps));

src/librustc/ty/structural_impls.rs

-17
Original file line numberDiff line numberDiff line change
@@ -908,23 +908,6 @@ impl<'tcx> TypeFoldable<'tcx> for ty::ClosureUpvar<'tcx> {
908908
}
909909
}
910910

911-
impl<'tcx> TypeFoldable<'tcx> for ty::ParameterEnvironment<'tcx> {
912-
fn super_fold_with<'gcx: 'tcx, F: TypeFolder<'gcx, 'tcx>>(&self, folder: &mut F) -> Self {
913-
ty::ParameterEnvironment {
914-
free_substs: self.free_substs.fold_with(folder),
915-
implicit_region_bound: self.implicit_region_bound.fold_with(folder),
916-
caller_bounds: self.caller_bounds.fold_with(folder),
917-
free_id_outlive: self.free_id_outlive,
918-
}
919-
}
920-
921-
fn super_visit_with<V: TypeVisitor<'tcx>>(&self, visitor: &mut V) -> bool {
922-
self.free_substs.visit_with(visitor) ||
923-
self.implicit_region_bound.visit_with(visitor) ||
924-
self.caller_bounds.visit_with(visitor)
925-
}
926-
}
927-
928911
impl<'tcx, T: TypeFoldable<'tcx>> TypeFoldable<'tcx> for ty::error::ExpectedFound<T> {
929912
fn super_fold_with<'gcx: 'tcx, F: TypeFolder<'gcx, 'tcx>>(&self, folder: &mut F) -> Self {
930913
ty::error::ExpectedFound {

src/librustc/ty/util.rs

+25-6
Original file line numberDiff line numberDiff line change
@@ -20,9 +20,11 @@ use ty::{Disr, ParameterEnvironment};
2020
use ty::fold::TypeVisitor;
2121
use ty::layout::{Layout, LayoutError};
2222
use ty::TypeVariants::*;
23+
use util::nodemap::FnvHashMap;
2324

2425
use rustc_const_math::{ConstInt, ConstIsize, ConstUsize};
2526

27+
use std::cell::RefCell;
2628
use std::cmp;
2729
use std::hash::{Hash, Hasher};
2830
use std::collections::hash_map::DefaultHasher;
@@ -579,11 +581,24 @@ impl<'a, 'gcx, 'tcx, H: Hasher> TypeVisitor<'tcx> for TypeIdHasher<'a, 'gcx, 'tc
579581
impl<'a, 'tcx> ty::TyS<'tcx> {
580582
fn impls_bound(&'tcx self, tcx: TyCtxt<'a, 'tcx, 'tcx>,
581583
param_env: &ParameterEnvironment<'tcx>,
582-
bound: ty::BuiltinBound, span: Span) -> bool
584+
bound: ty::BuiltinBound,
585+
cache: &RefCell<FnvHashMap<Ty<'tcx>, bool>>,
586+
span: Span) -> bool
583587
{
584-
tcx.infer_ctxt(None, Some(param_env.clone()), Reveal::ExactMatch).enter(|infcx| {
585-
traits::type_known_to_meet_builtin_bound(&infcx, self, bound, span)
586-
})
588+
if self.has_param_types() || self.has_self_ty() {
589+
if let Some(result) = cache.borrow().get(self) {
590+
return *result;
591+
}
592+
}
593+
let result =
594+
tcx.infer_ctxt(None, Some(param_env.clone()), Reveal::ExactMatch)
595+
.enter(|infcx| {
596+
traits::type_known_to_meet_builtin_bound(&infcx, self, bound, span)
597+
});
598+
if self.has_param_types() || self.has_self_ty() {
599+
cache.borrow_mut().insert(self, result);
600+
}
601+
return result;
587602
}
588603

589604
// FIXME (@jroesch): I made this public to use it, not sure if should be private
@@ -610,7 +625,9 @@ impl<'a, 'tcx> ty::TyS<'tcx> {
610625
TyArray(..) | TySlice(..) | TyTrait(..) | TyTuple(..) |
611626
TyClosure(..) | TyAdt(..) | TyAnon(..) |
612627
TyProjection(..) | TyParam(..) | TyInfer(..) | TyError => None
613-
}.unwrap_or_else(|| !self.impls_bound(tcx, param_env, ty::BoundCopy, span));
628+
}.unwrap_or_else(|| {
629+
!self.impls_bound(tcx, param_env, ty::BoundCopy, &param_env.is_copy_cache, span)
630+
});
614631

615632
if !self.has_param_types() && !self.has_self_ty() {
616633
self.flags.set(self.flags.get() | if result {
@@ -650,7 +667,9 @@ impl<'a, 'tcx> ty::TyS<'tcx> {
650667

651668
TyAdt(..) | TyProjection(..) | TyParam(..) |
652669
TyInfer(..) | TyAnon(..) | TyError => None
653-
}.unwrap_or_else(|| self.impls_bound(tcx, param_env, ty::BoundSized, span));
670+
}.unwrap_or_else(|| {
671+
self.impls_bound(tcx, param_env, ty::BoundSized, &param_env.is_sized_cache, span)
672+
});
654673

655674
if !self.has_param_types() && !self.has_self_ty() {
656675
self.flags.set(self.flags.get() | if result {

src/librustc_typeck/check/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1458,7 +1458,7 @@ impl<'a, 'gcx, 'tcx> FnCtxt<'a, 'gcx, 'tcx> {
14581458
}
14591459
}
14601460

1461-
pub fn param_env(&self) -> &ty::ParameterEnvironment<'tcx> {
1461+
pub fn param_env(&self) -> &ty::ParameterEnvironment<'gcx> {
14621462
&self.parameter_environment
14631463
}
14641464

0 commit comments

Comments
 (0)