blob: 11ae1531c4d1ab1dad7284dff6405fe4a7796ed8 [file] [log] [blame]
//
// Copyright (C) 2023 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
//! Trace provider backed by Intel LBR, using simpleperf tool.
use anyhow::{anyhow, Result};
use std::fs::{read_dir, remove_file};
use std::path::{Path, PathBuf};
use std::time::Duration;
use trace_provider::TraceProvider;
use crate::trace_provider;
static LBR_TRACEFILE_EXTENSION: &str = "lbrtrace";
static LBR_PROFILE_EXTENSION: &str = "data";
// Use a prime value to make sure that there are no weird interactions with e.g. short loops.
static LBR_SAMPLE_PERIOD: &str = "500009";
pub struct SimpleperfLbrTraceProvider {}
impl TraceProvider for SimpleperfLbrTraceProvider {
fn get_name(&self) -> &'static str {
"simpleperf_lbr"
}
fn is_ready(&self) -> bool {
true
}
fn trace_system(
&self,
trace_dir: &Path,
tag: &str,
sampling_period: &Duration,
binary_filter: &str,
) {
let trace_file = trace_provider::get_path(trace_dir, tag, LBR_TRACEFILE_EXTENSION);
// Record ETM data for kernel space only when it's not filtered out by binary_filter. So we
// can get more ETM data for user space when ETM data for kernel space isn't needed.
let event_name = if binary_filter.contains("kernel") {
"BR_INST_RETIRED.NEAR_TAKEN"
} else {
"BR_INST_RETIRED.NEAR_TAKEN:u"
};
let duration: String = sampling_period.as_secs_f64().to_string();
let args: Vec<&str> = vec![
"-a",
"-e",
event_name,
"-c",
LBR_SAMPLE_PERIOD,
"--duration",
&duration,
"-b",
"--exclude-perf",
"--binary",
binary_filter,
"--no-dump-symbols",
"--no-dump-kernel-symbols",
"-o",
trace_file.to_str().unwrap(),
];
simpleperf_profcollect::run_record_cmd(&args);
}
fn trace_process(
&self,
trace_dir: &Path,
tag: &str,
sampling_period: &Duration,
processes: &str,
) {
let trace_file = trace_provider::get_path(trace_dir, tag, LBR_TRACEFILE_EXTENSION);
let event_name = "BR_INST_RETIRED.NEAR_TAKEN:u";
let duration: String = sampling_period.as_secs_f64().to_string();
let args: Vec<&str> = vec![
"-p",
processes,
"-e",
event_name,
"-c",
LBR_SAMPLE_PERIOD,
"--duration",
&duration,
"-b",
"--no-dump-symbols",
"-o",
trace_file.to_str().unwrap(),
];
simpleperf_profcollect::run_record_cmd(&args);
}
fn process(&self, trace_dir: &Path, profile_dir: &Path, binary_filter: &str) -> Result<()> {
let is_lbr_extension = |file: &PathBuf| {
file.extension()
.and_then(|f| f.to_str())
.filter(|ext| ext == &LBR_TRACEFILE_EXTENSION)
.is_some()
};
let process_trace_file = |trace_file: PathBuf| {
let mut profile_file = PathBuf::from(profile_dir);
profile_file.push(
trace_file
.file_name()
.ok_or_else(|| anyhow!("Malformed trace path: {}", trace_file.display()))?,
);
profile_file.set_extension(LBR_PROFILE_EXTENSION);
let args: Vec<&str> = vec![
"-i",
trace_file.to_str().unwrap(),
"-o",
profile_file.to_str().unwrap(),
"--output",
"branch-list",
"--binary",
binary_filter,
];
simpleperf_profcollect::run_inject_cmd(&args);
remove_file(&trace_file)?;
Ok(())
};
read_dir(trace_dir)?
.filter_map(|e| e.ok())
.map(|e| e.path())
.filter(|e| e.is_file())
.filter(is_lbr_extension)
.try_for_each(process_trace_file)
}
fn set_log_file(&self, filename: &Path) {
simpleperf_profcollect::set_log_file(filename);
}
fn reset_log_file(&self) {
simpleperf_profcollect::reset_log_file();
}
}
impl SimpleperfLbrTraceProvider {
pub fn supported() -> bool {
simpleperf_profcollect::is_lbr_available()
}
}