Add support for artifact size profiling
This commit is contained in:
parent
ca8078d7b2
commit
947a33bf20
8 changed files with 76 additions and 16 deletions
25
Cargo.lock
25
Cargo.lock
|
@ -2138,6 +2138,19 @@ dependencies = [
|
||||||
"smallvec",
|
"smallvec",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "measureme"
|
||||||
|
version = "9.1.2"
|
||||||
|
source = "git+https://github.com/rylev/measureme#b9cccd7ad4c859a5e0e3dd6bff3daac7a190bdd7"
|
||||||
|
dependencies = [
|
||||||
|
"log",
|
||||||
|
"memmap2",
|
||||||
|
"parking_lot",
|
||||||
|
"perf-event-open-sys",
|
||||||
|
"rustc-hash",
|
||||||
|
"smallvec",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "memchr"
|
name = "memchr"
|
||||||
version = "2.4.1"
|
version = "2.4.1"
|
||||||
|
@ -2242,8 +2255,8 @@ dependencies = [
|
||||||
"hex 0.4.2",
|
"hex 0.4.2",
|
||||||
"libc",
|
"libc",
|
||||||
"log",
|
"log",
|
||||||
"measureme",
|
"measureme 9.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"rand 0.8.4",
|
"rand 0.8.3",
|
||||||
"rustc-workspace-hack",
|
"rustc-workspace-hack",
|
||||||
"rustc_version 0.4.0",
|
"rustc_version 0.4.0",
|
||||||
"shell-escape",
|
"shell-escape",
|
||||||
|
@ -3219,7 +3232,7 @@ dependencies = [
|
||||||
"indexmap",
|
"indexmap",
|
||||||
"jobserver",
|
"jobserver",
|
||||||
"libc",
|
"libc",
|
||||||
"measureme",
|
"measureme 9.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"memmap2",
|
"memmap2",
|
||||||
"parking_lot",
|
"parking_lot",
|
||||||
"rustc-ap-rustc_graphviz",
|
"rustc-ap-rustc_graphviz",
|
||||||
|
@ -3657,7 +3670,7 @@ dependencies = [
|
||||||
"bitflags",
|
"bitflags",
|
||||||
"cstr",
|
"cstr",
|
||||||
"libc",
|
"libc",
|
||||||
"measureme",
|
"measureme 9.1.2 (git+https://github.com/rylev/measureme)",
|
||||||
"rustc-demangle",
|
"rustc-demangle",
|
||||||
"rustc_arena",
|
"rustc_arena",
|
||||||
"rustc_ast",
|
"rustc_ast",
|
||||||
|
@ -3752,7 +3765,7 @@ dependencies = [
|
||||||
"indexmap",
|
"indexmap",
|
||||||
"jobserver",
|
"jobserver",
|
||||||
"libc",
|
"libc",
|
||||||
"measureme",
|
"measureme 9.1.2 (git+https://github.com/rylev/measureme)",
|
||||||
"memmap2",
|
"memmap2",
|
||||||
"parking_lot",
|
"parking_lot",
|
||||||
"rustc-hash",
|
"rustc-hash",
|
||||||
|
@ -4276,7 +4289,7 @@ dependencies = [
|
||||||
name = "rustc_query_impl"
|
name = "rustc_query_impl"
|
||||||
version = "0.0.0"
|
version = "0.0.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"measureme",
|
"measureme 9.1.2 (git+https://github.com/rylev/measureme)",
|
||||||
"rustc-rayon-core",
|
"rustc-rayon-core",
|
||||||
"rustc_ast",
|
"rustc_ast",
|
||||||
"rustc_data_structures",
|
"rustc_data_structures",
|
||||||
|
|
|
@ -11,7 +11,7 @@ doctest = false
|
||||||
bitflags = "1.0"
|
bitflags = "1.0"
|
||||||
cstr = "0.2"
|
cstr = "0.2"
|
||||||
libc = "0.2"
|
libc = "0.2"
|
||||||
measureme = "9.1.0"
|
measureme = { git = "https://github.com/rylev/measureme" }
|
||||||
snap = "1"
|
snap = "1"
|
||||||
tracing = "0.1"
|
tracing = "0.1"
|
||||||
rustc_middle = { path = "../rustc_middle" }
|
rustc_middle = { path = "../rustc_middle" }
|
||||||
|
|
|
@ -23,7 +23,7 @@ rustc-hash = "1.1.0"
|
||||||
smallvec = { version = "1.6.1", features = ["union", "may_dangle"] }
|
smallvec = { version = "1.6.1", features = ["union", "may_dangle"] }
|
||||||
rustc_index = { path = "../rustc_index", package = "rustc_index" }
|
rustc_index = { path = "../rustc_index", package = "rustc_index" }
|
||||||
bitflags = "1.2.1"
|
bitflags = "1.2.1"
|
||||||
measureme = "9.1.0"
|
measureme = { git = "https://github.com/rylev/measureme" }
|
||||||
libc = "0.2"
|
libc = "0.2"
|
||||||
stacker = "0.1.14"
|
stacker = "0.1.14"
|
||||||
tempfile = "3.2"
|
tempfile = "3.2"
|
||||||
|
|
|
@ -110,12 +110,14 @@ bitflags::bitflags! {
|
||||||
const FUNCTION_ARGS = 1 << 6;
|
const FUNCTION_ARGS = 1 << 6;
|
||||||
const LLVM = 1 << 7;
|
const LLVM = 1 << 7;
|
||||||
const INCR_RESULT_HASHING = 1 << 8;
|
const INCR_RESULT_HASHING = 1 << 8;
|
||||||
|
const ARTIFACT_SIZES = 1 << 9;
|
||||||
|
|
||||||
const DEFAULT = Self::GENERIC_ACTIVITIES.bits |
|
const DEFAULT = Self::GENERIC_ACTIVITIES.bits |
|
||||||
Self::QUERY_PROVIDERS.bits |
|
Self::QUERY_PROVIDERS.bits |
|
||||||
Self::QUERY_BLOCKED.bits |
|
Self::QUERY_BLOCKED.bits |
|
||||||
Self::INCR_CACHE_LOADS.bits |
|
Self::INCR_CACHE_LOADS.bits |
|
||||||
Self::INCR_RESULT_HASHING.bits;
|
Self::INCR_RESULT_HASHING.bits |
|
||||||
|
Self::ARTIFACT_SIZES.bits;
|
||||||
|
|
||||||
const ARGS = Self::QUERY_KEYS.bits | Self::FUNCTION_ARGS.bits;
|
const ARGS = Self::QUERY_KEYS.bits | Self::FUNCTION_ARGS.bits;
|
||||||
}
|
}
|
||||||
|
@ -136,6 +138,7 @@ const EVENT_FILTERS_BY_NAME: &[(&str, EventFilter)] = &[
|
||||||
("args", EventFilter::ARGS),
|
("args", EventFilter::ARGS),
|
||||||
("llvm", EventFilter::LLVM),
|
("llvm", EventFilter::LLVM),
|
||||||
("incr-result-hashing", EventFilter::INCR_RESULT_HASHING),
|
("incr-result-hashing", EventFilter::INCR_RESULT_HASHING),
|
||||||
|
("artifact-sizes", EventFilter::ARTIFACT_SIZES),
|
||||||
];
|
];
|
||||||
|
|
||||||
/// Something that uniquely identifies a query invocation.
|
/// Something that uniquely identifies a query invocation.
|
||||||
|
@ -285,6 +288,33 @@ impl SelfProfilerRef {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Record the size of an artifact that the compiler produces
|
||||||
|
///
|
||||||
|
/// `artifact_kind` is the class of artifact (e.g., query_cache, object_file, etc.)
|
||||||
|
/// `artifact_name` is an identifier to the specific artifact being stored (usually a filename)
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn artifact_size<A>(&self, artifact_kind: &str, artifact_name: A, size: u64)
|
||||||
|
where
|
||||||
|
A: Borrow<str> + Into<String>,
|
||||||
|
{
|
||||||
|
drop(self.exec(EventFilter::ARTIFACT_SIZES, |profiler| {
|
||||||
|
let builder = EventIdBuilder::new(&profiler.profiler);
|
||||||
|
let event_label = profiler.get_or_alloc_cached_string(artifact_kind);
|
||||||
|
let event_arg = profiler.get_or_alloc_cached_string(artifact_name);
|
||||||
|
let event_id = builder.from_label_and_arg(event_label, event_arg);
|
||||||
|
let thread_id = get_thread_id();
|
||||||
|
|
||||||
|
profiler.profiler.record_integer_event(
|
||||||
|
profiler.artifact_size_event_kind,
|
||||||
|
event_id,
|
||||||
|
thread_id,
|
||||||
|
size,
|
||||||
|
);
|
||||||
|
|
||||||
|
TimingGuard::none()
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
#[inline(always)]
|
#[inline(always)]
|
||||||
pub fn generic_activity_with_args(
|
pub fn generic_activity_with_args(
|
||||||
&self,
|
&self,
|
||||||
|
@ -372,7 +402,7 @@ impl SelfProfilerRef {
|
||||||
) {
|
) {
|
||||||
drop(self.exec(event_filter, |profiler| {
|
drop(self.exec(event_filter, |profiler| {
|
||||||
let event_id = StringId::new_virtual(query_invocation_id.0);
|
let event_id = StringId::new_virtual(query_invocation_id.0);
|
||||||
let thread_id = std::thread::current().id().as_u64().get() as u32;
|
let thread_id = get_thread_id();
|
||||||
|
|
||||||
profiler.profiler.record_instant_event(
|
profiler.profiler.record_instant_event(
|
||||||
event_kind(profiler),
|
event_kind(profiler),
|
||||||
|
@ -425,6 +455,7 @@ pub struct SelfProfiler {
|
||||||
incremental_result_hashing_event_kind: StringId,
|
incremental_result_hashing_event_kind: StringId,
|
||||||
query_blocked_event_kind: StringId,
|
query_blocked_event_kind: StringId,
|
||||||
query_cache_hit_event_kind: StringId,
|
query_cache_hit_event_kind: StringId,
|
||||||
|
artifact_size_event_kind: StringId,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SelfProfiler {
|
impl SelfProfiler {
|
||||||
|
@ -447,6 +478,7 @@ impl SelfProfiler {
|
||||||
profiler.alloc_string("IncrementalResultHashing");
|
profiler.alloc_string("IncrementalResultHashing");
|
||||||
let query_blocked_event_kind = profiler.alloc_string("QueryBlocked");
|
let query_blocked_event_kind = profiler.alloc_string("QueryBlocked");
|
||||||
let query_cache_hit_event_kind = profiler.alloc_string("QueryCacheHit");
|
let query_cache_hit_event_kind = profiler.alloc_string("QueryCacheHit");
|
||||||
|
let artifact_size_event_kind = profiler.alloc_string("ArtifactSize");
|
||||||
|
|
||||||
let mut event_filter_mask = EventFilter::empty();
|
let mut event_filter_mask = EventFilter::empty();
|
||||||
|
|
||||||
|
@ -491,6 +523,7 @@ impl SelfProfiler {
|
||||||
incremental_result_hashing_event_kind,
|
incremental_result_hashing_event_kind,
|
||||||
query_blocked_event_kind,
|
query_blocked_event_kind,
|
||||||
query_cache_hit_event_kind,
|
query_cache_hit_event_kind,
|
||||||
|
artifact_size_event_kind,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -561,7 +594,7 @@ impl<'a> TimingGuard<'a> {
|
||||||
event_kind: StringId,
|
event_kind: StringId,
|
||||||
event_id: EventId,
|
event_id: EventId,
|
||||||
) -> TimingGuard<'a> {
|
) -> TimingGuard<'a> {
|
||||||
let thread_id = std::thread::current().id().as_u64().get() as u32;
|
let thread_id = get_thread_id();
|
||||||
let raw_profiler = &profiler.profiler;
|
let raw_profiler = &profiler.profiler;
|
||||||
let timing_guard =
|
let timing_guard =
|
||||||
raw_profiler.start_recording_interval_event(event_kind, event_id, thread_id);
|
raw_profiler.start_recording_interval_event(event_kind, event_id, thread_id);
|
||||||
|
@ -655,6 +688,10 @@ pub fn duration_to_secs_str(dur: std::time::Duration) -> String {
|
||||||
format!("{:.3}", dur.as_secs_f64())
|
format!("{:.3}", dur.as_secs_f64())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_thread_id() -> u32 {
|
||||||
|
std::thread::current().id().as_u64().get() as u32
|
||||||
|
}
|
||||||
|
|
||||||
// Memory reporting
|
// Memory reporting
|
||||||
cfg_if! {
|
cfg_if! {
|
||||||
if #[cfg(windows)] {
|
if #[cfg(windows)] {
|
||||||
|
|
|
@ -95,6 +95,12 @@ where
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sess.prof.artifact_size(
|
||||||
|
&name.replace(' ', "_"),
|
||||||
|
path_buf.file_name().unwrap().to_string_lossy(),
|
||||||
|
encoder.position() as u64,
|
||||||
|
);
|
||||||
|
|
||||||
debug!("save: data written to disk successfully");
|
debug!("save: data written to disk successfully");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ edition = "2021"
|
||||||
doctest = false
|
doctest = false
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
measureme = "9.0.0"
|
measureme = { git = "https://github.com/rylev/measureme" }
|
||||||
rustc-rayon-core = "0.3.1"
|
rustc-rayon-core = "0.3.1"
|
||||||
tracing = "0.1"
|
tracing = "0.1"
|
||||||
rustc_ast = { path = "../rustc_ast" }
|
rustc_ast = { path = "../rustc_ast" }
|
||||||
|
|
|
@ -222,7 +222,7 @@ impl<K: DepKind> EncoderState<K> {
|
||||||
index
|
index
|
||||||
}
|
}
|
||||||
|
|
||||||
fn finish(self) -> FileEncodeResult {
|
fn finish(self, profiler: &SelfProfilerRef) -> FileEncodeResult {
|
||||||
let Self { mut encoder, total_node_count, total_edge_count, result, stats: _ } = self;
|
let Self { mut encoder, total_node_count, total_edge_count, result, stats: _ } = self;
|
||||||
let () = result?;
|
let () = result?;
|
||||||
|
|
||||||
|
@ -235,7 +235,11 @@ impl<K: DepKind> EncoderState<K> {
|
||||||
IntEncodedWithFixedSize(edge_count).encode(&mut encoder)?;
|
IntEncodedWithFixedSize(edge_count).encode(&mut encoder)?;
|
||||||
debug!("position: {:?}", encoder.position());
|
debug!("position: {:?}", encoder.position());
|
||||||
// Drop the encoder so that nothing is written after the counts.
|
// Drop the encoder so that nothing is written after the counts.
|
||||||
encoder.flush()
|
let result = encoder.flush();
|
||||||
|
// FIXME(rylev): we hardcode the dep graph file name so we don't need a dependency on
|
||||||
|
// rustc_incremental just for that.
|
||||||
|
profiler.artifact_size("dep_graph", "dep-graph.bin", encoder.position() as u64);
|
||||||
|
result
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -332,6 +336,6 @@ impl<K: DepKind + Encodable<FileEncoder>> GraphEncoder<K> {
|
||||||
|
|
||||||
pub fn finish(self, profiler: &SelfProfilerRef) -> FileEncodeResult {
|
pub fn finish(self, profiler: &SelfProfilerRef) -> FileEncodeResult {
|
||||||
let _prof_timer = profiler.generic_activity("incr_comp_encode_dep_graph");
|
let _prof_timer = profiler.generic_activity("incr_comp_encode_dep_graph");
|
||||||
self.status.into_inner().finish()
|
self.status.into_inner().finish(profiler)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1279,7 +1279,7 @@ options! {
|
||||||
"specify the events recorded by the self profiler;
|
"specify the events recorded by the self profiler;
|
||||||
for example: `-Z self-profile-events=default,query-keys`
|
for example: `-Z self-profile-events=default,query-keys`
|
||||||
all options: none, all, default, generic-activity, query-provider, query-cache-hit
|
all options: none, all, default, generic-activity, query-provider, query-cache-hit
|
||||||
query-blocked, incr-cache-load, incr-result-hashing, query-keys, function-args, args, llvm"),
|
query-blocked, incr-cache-load, incr-result-hashing, query-keys, function-args, args, llvm, artifact-sizes"),
|
||||||
share_generics: Option<bool> = (None, parse_opt_bool, [TRACKED],
|
share_generics: Option<bool> = (None, parse_opt_bool, [TRACKED],
|
||||||
"make the current crate share its generic instantiations"),
|
"make the current crate share its generic instantiations"),
|
||||||
show_span: Option<String> = (None, parse_opt_string, [TRACKED],
|
show_span: Option<String> = (None, parse_opt_string, [TRACKED],
|
||||||
|
|
Loading…
Add table
Reference in a new issue