Skip to content

Commit 20f2601

Browse files
authored
Rollup merge of #112155 - nnethercote:debug_dump, r=wesleywiser
Improve CGU debug printing. - Add more total and per-CGU measurements. - Ensure CGUs are sorted by name before the first `debug_dump` calls, for deterministic output. - Print items within CGUs in sorted-by-name order, for deterministic output. - Add some assertions and comments clarifying sortedness of CGUs at various points. An example, before: ``` INITIAL PARTITIONING (5 CodegenUnits, max=29, min=1, max/min=29.0): CodegenUnit scev95ysd7g4b0z estimated size 2: - fn <() as std::process::Termination>::report [(External, Hidden)] [h082b15a6d07338dcE] estimated size 2 CodegenUnit 1j0frgtl72rsz24q estimated size 29: - fn std::rt::lang_start::<()>::{closure#0} [(External, Hidden)] [h695c7b5d6a212565E] estimated size 17 - fn std::rt::lang_start::<()> [(External, Hidden)] [h4ca942948e9cb931E] estimated size 12 CodegenUnit 5dbzi1e5qm0d7kj2 estimated size 4: - fn <[closure@std::rt::lang_start<()>::{closure#0}] as std::ops::FnOnce<()>>::call_once - shim [(External, Hidden)] [h24eaa44f03b2b233E] estimated size 1 - fn <fn() as std::ops::FnOnce<()>>::call_once - shim(fn()) [(External, Hidden)] [hf338f5339c3711acE] estimated size 1 - fn <[closure@std::rt::lang_start<()>::{closure#0}] as std::ops::FnOnce<()>>::call_once - shim(vtable) [(External, Hidden)] [h595d414cbb7651d5E] estimated size 1 - fn std::ptr::drop_in_place::<[closure@std::rt::lang_start<()>::{closure#0}]> - shim(None) [(External, Hidden)] [h17a19dcdb40600daE] estimated size 1 CodegenUnit 220m1mqa2mlbg7r3 estimated size 1: - fn main [(External, Hidden)] [hb29587cdb6db5f42E] estimated size 1 CodegenUnit 4ulbh241f7tvyn7x estimated size 6: - fn std::sys_common::backtrace::__rust_begin_short_backtrace::<fn(), ()> [(External, Hidden)] [h41dada2c21a1259dE] estimated size 6 ``` and after: ``` INITIAL PARTITIONING (9 items, total_size=42; 5 CGUs, max_size=29, min_size=1, max_size/min_size=29.0): - CGU[0] 1j0frgtl72rsz24q (2 items, size=29): - fn std::rt::lang_start::<()> [(External, Hidden)] [h4ca942948e9cb931E] (size=12) - fn std::rt::lang_start::<()>::{closure#0} [(External, Hidden)] [h695c7b5d6a212565E] (size=17) - CGU[1] 220m1mqa2mlbg7r3 (1 items, size=1): - fn main [(External, Hidden)] [hb29587cdb6db5f42E] (size=1) - CGU[2] 4ulbh241f7tvyn7x (1 items, size=6): - fn std::sys_common::backtrace::__rust_begin_short_backtrace::<fn(), ()> [(External, Hidden)] [h41dada2c21a1259dE] (size=6) - CGU[3] 5dbzi1e5qm0d7kj2 (4 items, size=4): - fn <[closure@std::rt::lang_start<()>::{closure#0}] as std::ops::FnOnce<()>>::call_once - shim(vtable) [(External, Hidden)] [h595d414cbb7651d5E] (size=1) - fn <[closure@std::rt::lang_start<()>::{closure#0}] as std::ops::FnOnce<()>>::call_once - shim [(External, Hidden)] [h24eaa44f03b2b233E] (size=1) - fn <fn() as std::ops::FnOnce<()>>::call_once - shim(fn()) [(External, Hidden)] [hf338f5339c3711acE] (size=1) - fn std::ptr::drop_in_place::<[closure@std::rt::lang_start<()>::{closure#0}]> - shim(None) [(External, Hidden)] [h17a19dcdb40600daE] (size=1) - CGU[4] scev95ysd7g4b0z (1 items, size=2): - fn <() as std::process::Termination>::report [(External, Hidden)] [h082b15a6d07338dcE] (size=2) ``` r? `@wesleywiser`
2 parents 981fe09 + 1191bea commit 20f2601

File tree

3 files changed

+43
-28
lines changed

3 files changed

+43
-28
lines changed

compiler/rustc_middle/src/mir/mono.rs

+2
Original file line numberDiff line numberDiff line change
@@ -291,10 +291,12 @@ impl<'tcx> CodegenUnit<'tcx> {
291291
self.primary = true;
292292
}
293293

294+
/// The order of these items is non-determinstic.
294295
pub fn items(&self) -> &FxHashMap<MonoItem<'tcx>, (Linkage, Visibility)> {
295296
&self.items
296297
}
297298

299+
/// The order of these items is non-determinstic.
298300
pub fn items_mut(&mut self) -> &mut FxHashMap<MonoItem<'tcx>, (Linkage, Visibility)> {
299301
&mut self.items
300302
}

compiler/rustc_monomorphize/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
#![feature(array_windows)]
2+
#![feature(is_sorted)]
23
#![recursion_limit = "256"]
34
#![allow(rustc::potential_query_instability)]
45
#![deny(rustc::untranslatable_diagnostic)]

compiler/rustc_monomorphize/src/partitioning.rs

+40-28
Original file line numberDiff line numberDiff line change
@@ -126,11 +126,14 @@ struct PartitioningCx<'a, 'tcx> {
126126
}
127127

128128
struct PlacedRootMonoItems<'tcx> {
129+
/// The codegen units, sorted by name to make things deterministic.
129130
codegen_units: Vec<CodegenUnit<'tcx>>,
131+
130132
roots: FxHashSet<MonoItem<'tcx>>,
131133
internalization_candidates: FxHashSet<MonoItem<'tcx>>,
132134
}
133135

136+
// The output CGUs are sorted by name.
134137
fn partition<'tcx, I>(
135138
tcx: TyCtxt<'tcx>,
136139
mono_items: &mut I,
@@ -143,6 +146,7 @@ where
143146
let _prof_timer = tcx.prof.generic_activity("cgu_partitioning");
144147

145148
let cx = &PartitioningCx { tcx, target_cgu_count: max_cgu_count, inlining_map };
149+
146150
// In the first step, we place all regular monomorphizations into their
147151
// respective 'home' codegen unit. Regular monomorphizations are all
148152
// functions and statics defined in the local crate.
@@ -225,8 +229,8 @@ where
225229
dead_code_cgu.make_code_coverage_dead_code_cgu();
226230
}
227231

228-
// Finally, sort by codegen unit name, so that we get deterministic results.
229-
codegen_units.sort_by(|a, b| a.name().as_str().cmp(b.name().as_str()));
232+
// Ensure CGUs are sorted by name, so that we get deterministic results.
233+
assert!(codegen_units.is_sorted_by(|a, b| Some(a.name().as_str().cmp(b.name().as_str()))));
230234

231235
debug_dump(tcx, "FINAL", &codegen_units);
232236

@@ -301,27 +305,22 @@ where
301305
codegen_units.insert(codegen_unit_name, CodegenUnit::new(codegen_unit_name));
302306
}
303307

304-
let codegen_units = codegen_units.into_values().collect();
308+
let mut codegen_units: Vec<_> = codegen_units.into_values().collect();
309+
codegen_units.sort_by(|a, b| a.name().as_str().cmp(b.name().as_str()));
310+
305311
PlacedRootMonoItems { codegen_units, roots, internalization_candidates }
306312
}
307313

314+
// This function requires the CGUs to be sorted by name on input, and ensures
315+
// they are sorted by name on return, for deterministic behaviour.
308316
fn merge_codegen_units<'tcx>(
309317
cx: &PartitioningCx<'_, 'tcx>,
310318
codegen_units: &mut Vec<CodegenUnit<'tcx>>,
311319
) {
312320
assert!(cx.target_cgu_count >= 1);
313321

314-
// Note that at this point in time the `codegen_units` here may not be
315-
// in a deterministic order (but we know they're deterministically the
316-
// same set). We want this merging to produce a deterministic ordering
317-
// of codegen units from the input.
318-
//
319-
// Due to basically how we've implemented the merging below (merge the
320-
// two smallest into each other) we're sure to start off with a
321-
// deterministic order (sorted by name). This'll mean that if two cgus
322-
// have the same size the stable sort below will keep everything nice
323-
// and deterministic.
324-
codegen_units.sort_by(|a, b| a.name().as_str().cmp(b.name().as_str()));
322+
// A sorted order here ensures merging is deterministic.
323+
assert!(codegen_units.is_sorted_by(|a, b| Some(a.name().as_str().cmp(b.name().as_str()))));
325324

326325
// This map keeps track of what got merged into what.
327326
let mut cgu_contents: FxHashMap<Symbol, Vec<Symbol>> =
@@ -400,6 +399,9 @@ fn merge_codegen_units<'tcx>(
400399
cgu.set_name(numbered_codegen_unit_name);
401400
}
402401
}
402+
403+
// A sorted order here ensures what follows can be deterministic.
404+
codegen_units.sort_by(|a, b| a.name().as_str().cmp(b.name().as_str()));
403405
}
404406

405407
/// For symbol internalization, we need to know whether a symbol/mono-item is
@@ -859,36 +861,46 @@ fn default_visibility(tcx: TyCtxt<'_>, id: DefId, is_generic: bool) -> Visibilit
859861
_ => Visibility::Hidden,
860862
}
861863
}
864+
862865
fn debug_dump<'a, 'tcx: 'a>(tcx: TyCtxt<'tcx>, label: &str, cgus: &[CodegenUnit<'tcx>]) {
863866
let dump = move || {
864867
use std::fmt::Write;
865868

866869
let num_cgus = cgus.len();
867-
let max = cgus.iter().map(|cgu| cgu.size_estimate()).max().unwrap();
868-
let min = cgus.iter().map(|cgu| cgu.size_estimate()).min().unwrap();
869-
let ratio = max as f64 / min as f64;
870+
let num_items: usize = cgus.iter().map(|cgu| cgu.items().len()).sum();
871+
let total_size: usize = cgus.iter().map(|cgu| cgu.size_estimate()).sum();
872+
let max_size = cgus.iter().map(|cgu| cgu.size_estimate()).max().unwrap();
873+
let min_size = cgus.iter().map(|cgu| cgu.size_estimate()).min().unwrap();
874+
let max_min_size_ratio = max_size as f64 / min_size as f64;
870875

871876
let s = &mut String::new();
872877
let _ = writeln!(
873878
s,
874-
"{label} ({num_cgus} CodegenUnits, max={max}, min={min}, max/min={ratio:.1}):"
879+
"{label} ({num_items} items, total_size={total_size}; {num_cgus} CGUs, \
880+
max_size={max_size}, min_size={min_size}, max_size/min_size={max_min_size_ratio:.1}):"
875881
);
876-
for cgu in cgus {
877-
let _ =
878-
writeln!(s, "CodegenUnit {} estimated size {}:", cgu.name(), cgu.size_estimate());
882+
for (i, cgu) in cgus.iter().enumerate() {
883+
let num_items = cgu.items().len();
884+
let _ = writeln!(
885+
s,
886+
"- CGU[{i}] {} ({num_items} items, size={}):",
887+
cgu.name(),
888+
cgu.size_estimate()
889+
);
879890

880-
for (mono_item, linkage) in cgu.items() {
881-
let symbol_name = mono_item.symbol_name(tcx).name;
891+
// The order of `cgu.items()` is non-deterministic; sort it by name
892+
// to give deterministic output.
893+
let mut items: Vec<_> = cgu.items().iter().collect();
894+
items.sort_by_key(|(item, _)| item.symbol_name(tcx).name);
895+
for (item, linkage) in items {
896+
let symbol_name = item.symbol_name(tcx).name;
882897
let symbol_hash_start = symbol_name.rfind('h');
883898
let symbol_hash = symbol_hash_start.map_or("<no hash>", |i| &symbol_name[i..]);
884899

900+
let size = item.size_estimate(tcx);
885901
let _ = with_no_trimmed_paths!(writeln!(
886902
s,
887-
" - {} [{:?}] [{}] estimated size {}",
888-
mono_item,
889-
linkage,
890-
symbol_hash,
891-
mono_item.size_estimate(tcx)
903+
" - {item} [{linkage:?}] [{symbol_hash}] (size={size})"
892904
));
893905
}
894906

0 commit comments

Comments
 (0)