2020-07-30 08:43:47 +00:00
|
|
|
//! Like `std::time::Instant`, but also measures memory & CPU cycles.
|
2020-07-30 07:44:21 +00:00
|
|
|
use std::{
|
|
|
|
fmt,
|
|
|
|
time::{Duration, Instant},
|
|
|
|
};
|
|
|
|
|
2020-07-30 08:40:55 +00:00
|
|
|
use crate::MemoryUsage;
|
|
|
|
|
2020-07-30 07:44:21 +00:00
|
|
|
pub struct StopWatch {
|
|
|
|
time: Instant,
|
2020-07-30 08:40:55 +00:00
|
|
|
#[cfg(target_os = "linux")]
|
2020-07-30 07:44:21 +00:00
|
|
|
counter: Option<perf_event::Counter>,
|
|
|
|
memory: Option<MemoryUsage>,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct StopWatchSpan {
|
|
|
|
pub time: Duration,
|
|
|
|
pub instructions: Option<u64>,
|
|
|
|
pub memory: Option<MemoryUsage>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl StopWatch {
|
|
|
|
pub fn start() -> StopWatch {
|
2020-07-30 08:40:55 +00:00
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
let counter = {
|
2021-11-16 18:32:38 +00:00
|
|
|
// When debugging rust-analyzer using rr, the perf-related syscalls cause it to abort.
|
|
|
|
// We allow disabling perf by setting the env var `RA_DISABLE_PERF`.
|
|
|
|
|
|
|
|
use once_cell::sync::Lazy;
|
|
|
|
static PERF_ENABLED: Lazy<bool> =
|
|
|
|
Lazy::new(|| std::env::var_os("RA_DISABLE_PERF").is_none());
|
|
|
|
|
|
|
|
if *PERF_ENABLED {
|
|
|
|
let mut counter = perf_event::Builder::new()
|
|
|
|
.build()
|
|
|
|
.map_err(|err| eprintln!("Failed to create perf counter: {}", err))
|
|
|
|
.ok();
|
|
|
|
if let Some(counter) = &mut counter {
|
|
|
|
if let Err(err) = counter.enable() {
|
|
|
|
eprintln!("Failed to start perf counter: {}", err)
|
|
|
|
}
|
2020-07-30 12:27:19 +00:00
|
|
|
}
|
2021-11-16 18:32:38 +00:00
|
|
|
counter
|
|
|
|
} else {
|
|
|
|
None
|
2020-07-30 08:40:55 +00:00
|
|
|
}
|
|
|
|
};
|
2020-07-30 07:44:21 +00:00
|
|
|
let time = Instant::now();
|
2020-07-30 08:40:55 +00:00
|
|
|
StopWatch {
|
|
|
|
time,
|
|
|
|
#[cfg(target_os = "linux")]
|
|
|
|
counter,
|
|
|
|
memory: None,
|
|
|
|
}
|
2020-07-30 07:44:21 +00:00
|
|
|
}
|
|
|
|
pub fn memory(mut self, yes: bool) -> StopWatch {
|
|
|
|
if yes {
|
2021-05-22 13:53:47 +00:00
|
|
|
self.memory = Some(MemoryUsage::now());
|
2020-07-30 07:44:21 +00:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
pub fn elapsed(&mut self) -> StopWatchSpan {
|
|
|
|
let time = self.time.elapsed();
|
2020-07-30 08:40:55 +00:00
|
|
|
|
|
|
|
#[cfg(target_os = "linux")]
|
2020-07-30 12:27:19 +00:00
|
|
|
let instructions = self.counter.as_mut().and_then(|it| {
|
|
|
|
it.read().map_err(|err| eprintln!("Failed to read perf counter: {}", err)).ok()
|
|
|
|
});
|
2020-07-30 08:40:55 +00:00
|
|
|
#[cfg(not(target_os = "linux"))]
|
|
|
|
let instructions = None;
|
|
|
|
|
2021-05-22 13:53:47 +00:00
|
|
|
let memory = self.memory.map(|it| MemoryUsage::now() - it);
|
2020-07-30 07:44:21 +00:00
|
|
|
StopWatchSpan { time, instructions, memory }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for StopWatchSpan {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
write!(f, "{:.2?}", self.time)?;
|
|
|
|
if let Some(mut instructions) = self.instructions {
|
|
|
|
let mut prefix = "";
|
|
|
|
if instructions > 10000 {
|
|
|
|
instructions /= 1000;
|
2021-10-03 12:39:43 +00:00
|
|
|
prefix = "k";
|
2020-07-30 07:44:21 +00:00
|
|
|
}
|
|
|
|
if instructions > 10000 {
|
|
|
|
instructions /= 1000;
|
2021-10-03 12:39:43 +00:00
|
|
|
prefix = "m";
|
2020-07-30 07:44:21 +00:00
|
|
|
}
|
2021-01-06 15:16:04 +00:00
|
|
|
if instructions > 10000 {
|
|
|
|
instructions /= 1000;
|
2021-10-03 12:39:43 +00:00
|
|
|
prefix = "g";
|
2021-01-06 15:16:04 +00:00
|
|
|
}
|
|
|
|
write!(f, ", {}{}instr", instructions, prefix)?;
|
2020-07-30 07:44:21 +00:00
|
|
|
}
|
|
|
|
if let Some(memory) = self.memory {
|
|
|
|
write!(f, ", {}", memory)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|