2017-03-01 05:28:27 +08:00
|
|
|
use std::mem;
|
2017-02-26 10:50:20 +08:00
|
|
|
use std::vec::Vec;
|
|
|
|
use std::string::String;
|
|
|
|
use std::btree_map::BTreeMap;
|
2017-03-01 05:28:27 +08:00
|
|
|
use std::io::Write;
|
2017-02-26 10:50:20 +08:00
|
|
|
|
2017-03-03 17:14:43 +08:00
|
|
|
const ALIGNMENT: usize = 64;
|
|
|
|
|
2017-02-26 10:50:20 +08:00
|
|
|
#[derive(Debug)]
|
|
|
|
struct Entry {
|
2017-04-15 15:27:09 +08:00
|
|
|
trace: Vec<u8>,
|
|
|
|
padding_len: usize,
|
2017-04-06 02:34:08 +08:00
|
|
|
duration: u64
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Manager {
|
|
|
|
entries: BTreeMap<String, Entry>,
|
2017-04-15 15:27:09 +08:00
|
|
|
recording_name: String,
|
|
|
|
recording_trace: Vec<u8>
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Manager {
|
|
|
|
pub fn new() -> Manager {
|
|
|
|
Manager {
|
|
|
|
entries: BTreeMap::new(),
|
2017-04-15 15:27:09 +08:00
|
|
|
recording_name: String::new(),
|
|
|
|
recording_trace: Vec::new(),
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-15 15:27:09 +08:00
|
|
|
pub fn record_start(&mut self, name: &str) {
|
|
|
|
self.recording_name = String::from(name);
|
|
|
|
self.recording_trace = Vec::new();
|
|
|
|
|
|
|
|
// or we could needlessly OOM replacing a large trace
|
|
|
|
self.entries.remove(name);
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|
|
|
|
|
2017-04-15 15:26:18 +08:00
|
|
|
pub fn record_append(&mut self, data: &[u8]) {
|
2017-04-15 15:27:09 +08:00
|
|
|
self.recording_trace.write_all(data).unwrap();
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|
|
|
|
|
2017-04-15 15:27:09 +08:00
|
|
|
pub fn record_stop(&mut self, duration: u64) {
|
|
|
|
let mut trace = Vec::new();
|
|
|
|
mem::swap(&mut self.recording_trace, &mut trace);
|
|
|
|
trace.push(0);
|
|
|
|
let data_len = trace.len();
|
2017-02-26 10:50:20 +08:00
|
|
|
|
2017-03-03 17:14:43 +08:00
|
|
|
// Realign.
|
2017-04-15 15:27:09 +08:00
|
|
|
trace.reserve(ALIGNMENT - 1);
|
|
|
|
let padding = ALIGNMENT - trace.as_ptr() as usize % ALIGNMENT;
|
2017-03-03 17:14:43 +08:00
|
|
|
let padding = if padding == ALIGNMENT { 0 } else { padding };
|
|
|
|
for _ in 0..padding {
|
|
|
|
// Vec guarantees that this will not reallocate
|
2017-04-15 15:27:09 +08:00
|
|
|
trace.push(0)
|
2017-03-03 17:14:43 +08:00
|
|
|
}
|
|
|
|
for i in 1..data_len + 1 {
|
2017-04-15 15:27:09 +08:00
|
|
|
trace[data_len + padding - i] = trace[data_len - i]
|
2017-03-03 17:14:43 +08:00
|
|
|
}
|
2017-02-26 10:50:20 +08:00
|
|
|
|
2017-04-15 15:27:09 +08:00
|
|
|
let mut name = String::new();
|
|
|
|
mem::swap(&mut self.recording_name, &mut name);
|
|
|
|
self.entries.insert(name, Entry {
|
|
|
|
trace: trace,
|
|
|
|
padding_len: padding,
|
2017-04-06 02:34:08 +08:00
|
|
|
duration: duration
|
2017-02-26 10:50:20 +08:00
|
|
|
});
|
|
|
|
}
|
2017-03-01 05:28:27 +08:00
|
|
|
|
|
|
|
pub fn erase(&mut self, name: &str) {
|
|
|
|
self.entries.remove(name);
|
|
|
|
}
|
|
|
|
|
2017-04-06 02:34:08 +08:00
|
|
|
pub fn with_trace<F, R>(&self, name: &str, f: F) -> R
|
|
|
|
where F: FnOnce(Option<&[u8]>, u64) -> R {
|
|
|
|
match self.entries.get(name) {
|
2017-04-15 15:27:09 +08:00
|
|
|
Some(entry) => f(Some(&entry.trace[entry.padding_len..]), entry.duration),
|
2017-04-06 02:34:08 +08:00
|
|
|
None => f(None, 0)
|
|
|
|
}
|
2017-03-01 05:28:27 +08:00
|
|
|
}
|
2017-02-26 10:50:20 +08:00
|
|
|
}
|