Skip to content

Commit cc8da78

Browse files
committed
Auto merge of rust-lang#127288 - lqd:typelen-cache, r=compiler-errors
cache type sizes in type-size limit visitor This is basically rust-lang#125507 (comment) as lcnr can't open the PR now. Locally it reduces the `itertools` regression by quite a bit, to "only +50%" compared to nightly (that includes overhead from the local lack of artifact post-processing, and is just a data point to compare to the 10-20x timings without the cache). ```console Benchmark 1: cargo +stage1 build --release Time (mean ± σ): 2.721 s ± 0.009 s [User: 2.446 s, System: 0.325 s] Range (min … max): 2.710 s … 2.738 s 10 runs Benchmark 2: cargo +nightly build --release Time (mean ± σ): 1.784 s ± 0.005 s [User: 1.540 s, System: 0.279 s] Range (min … max): 1.778 s … 1.792 s 10 runs Summary cargo +nightly build --release ran 1.52 ± 0.01 times faster than cargo +stage1 build --release ``` On master, it's from 34s to the 2.7s above. r? compiler-errors
2 parents c422581 + 529a3f4 commit cc8da78

File tree

1 file changed

+16
-3
lines changed

1 file changed

+16
-3
lines changed

compiler/rustc_middle/src/ty/instance.rs

+16-3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ use crate::ty::{
55
self, EarlyBinder, GenericArgs, GenericArgsRef, Ty, TyCtxt, TypeFoldable, TypeSuperFoldable,
66
TypeSuperVisitable, TypeVisitable, TypeVisitableExt, TypeVisitor,
77
};
8+
use rustc_data_structures::fx::FxHashMap;
89
use rustc_errors::ErrorGuaranteed;
910
use rustc_hir as hir;
1011
use rustc_hir::def::Namespace;
@@ -388,21 +389,33 @@ impl<'tcx> InstanceKind<'tcx> {
388389
}
389390

390391
fn type_length<'tcx>(item: impl TypeVisitable<TyCtxt<'tcx>>) -> usize {
391-
struct Visitor {
392+
struct Visitor<'tcx> {
392393
type_length: usize,
394+
cache: FxHashMap<Ty<'tcx>, usize>,
393395
}
394-
impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for Visitor {
396+
impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for Visitor<'tcx> {
395397
fn visit_ty(&mut self, t: Ty<'tcx>) {
398+
if let Some(&value) = self.cache.get(&t) {
399+
self.type_length += value;
400+
return;
401+
}
402+
403+
let prev = self.type_length;
396404
self.type_length += 1;
397405
t.super_visit_with(self);
406+
407+
// We don't try to use the cache if the type is fairly small.
408+
if self.type_length > 16 {
409+
self.cache.insert(t, self.type_length - prev);
410+
}
398411
}
399412

400413
fn visit_const(&mut self, ct: ty::Const<'tcx>) {
401414
self.type_length += 1;
402415
ct.super_visit_with(self);
403416
}
404417
}
405-
let mut visitor = Visitor { type_length: 0 };
418+
let mut visitor = Visitor { type_length: 0, cache: Default::default() };
406419
item.visit_with(&mut visitor);
407420

408421
visitor.type_length

0 commit comments

Comments
 (0)