use crate::config::Config;
use crate::source_analysis::*;
use crate::traces::*;
use cargo::core::Workspace;
use gimli::*;
use log::{debug, trace};
use memmap::MmapOptions;
use object::{File as OFile, Object};
use rustc_demangle::demangle;
use std::collections::HashMap;
use std::fs::File;
use std::io;
use std::path::{Path, PathBuf};
type FuncDesc = (u64, u64, FunctionType, Option<String>);
#[derive(Clone, Copy, PartialEq)]
enum FunctionType {
Generated,
Test,
Standard,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd)]
pub enum LineType {
TestMain,
TestEntry(u64),
FunctionEntry(u64),
Statement,
Condition,
Unknown,
UnusedGeneric,
}
#[derive(Clone, PartialEq, Eq, Hash)]
struct SourceLocation {
pub path: PathBuf,
pub line: u64,
}
#[derive(Debug, Clone)]
pub struct TracerData {
pub trace_type: LineType,
pub address: Option<u64>,
pub length: u64,
pub fn_name: Option<String>,
}
fn generate_func_desc<R, Offset>(
die: &DebuggingInformationEntry<R, Offset>,
debug_str: &DebugStr<R>,
) -> Result<FuncDesc>
where
R: Reader<Offset = Offset>,
Offset: ReaderOffset,
{
let mut func_type = FunctionType::Standard;
let low = die.attr_value(DW_AT_low_pc)?;
let high = die.attr_value(DW_AT_high_pc)?;
let linkage = die.attr_value(DW_AT_linkage_name)?;
let fn_name = die.attr_value(DW_AT_name)?;
let fn_name: Option<String> = match fn_name {
Some(AttributeValue::DebugStrRef(offset)) => debug_str
.get_str(offset)
.and_then(|r| r.to_string().map(|s| s.to_string()))
.ok()
.map(|r| demangle(r.as_ref()).to_string()),
_ => None,
};
let low = match low {
Some(AttributeValue::Addr(x)) => x,
_ => 0u64,
};
let high = match high {
Some(AttributeValue::Udata(x)) => x,
_ => 0u64,
};
if let Some(AttributeValue::DebugStrRef(offset)) = linkage {
let name = debug_str
.get_str(offset)
.and_then(|r| r.to_string().map(|s| s.to_string()))
.unwrap_or_else(|_| "".into());
let name = demangle(name.as_ref()).to_string();
func_type = if name.contains("tests::") {
FunctionType::Test
} else if name.contains("__test::main") {
FunctionType::Generated
} else {
FunctionType::Standard
};
}
Ok((low, high, func_type, fn_name))
}
fn get_entry_points<R, Offset>(
debug_info: &CompilationUnitHeader<R, Offset>,
debug_abbrev: &Abbreviations,
debug_str: &DebugStr<R>,
) -> Vec<FuncDesc>
where
R: Reader<Offset = Offset>,
Offset: ReaderOffset,
{
let mut result: Vec<FuncDesc> = Vec::new();
let mut cursor = debug_info.entries(debug_abbrev);
let _ = cursor.next_entry();
while let Ok(Some((_, node))) = cursor.next_dfs() {
if node.tag() == DW_TAG_subprogram {
if let Ok(fd) = generate_func_desc(node, debug_str) {
result.push(fd);
}
}
}
result
}
fn get_addresses_from_program<R, Offset>(
prog: IncompleteLineProgram<R>,
debug_strs: &DebugStr<R>,
entries: &Vec<(u64, LineType, &Option<String>)>,
project: &Path,
result: &mut HashMap<SourceLocation, Vec<TracerData>>,
) -> Result<()>
where
R: Reader<Offset = Offset>,
Offset: ReaderOffset,
{
let get_string = |x: R| x.to_string().map(|y| y.to_string()).ok();
let (cprog, seq) = prog.sequences()?;
for s in seq {
let mut temp_map: HashMap<SourceLocation, TracerData> = HashMap::new();
let mut sm = cprog.resume_from(&s);
while let Ok(Some((header, &ln_row))) = sm.next_row() {
if !ln_row.is_stmt() || ln_row.line().is_none() {
continue;
}
if let Some(file) = ln_row.file(header) {
let mut path = project.to_path_buf();
if let Some(dir) = file.directory(header) {
if let Some(temp) = dir.string_value(debug_strs).and_then(get_string) {
path.push(temp);
}
}
if let Ok(p) = path.canonicalize() {
path = p;
}
let is_target = if path.is_relative() {
path.starts_with("target")
} else {
path.starts_with(project.join("target"))
};
if !is_target && path.starts_with(project) {
if let Some(file) = ln_row.file(header) {
let line = ln_row.line().unwrap();
let file = file.path_name();
if let Some(file) = file.string_value(debug_strs).and_then(get_string) {
path.push(file);
if !path.is_file() {
continue;
}
let address = ln_row.address();
let (desc, fn_name) = entries
.iter()
.filter(|&&(addr, _, _)| addr == address)
.map(|&(_, t, fn_name)| (t, fn_name.to_owned()))
.nth(0)
.unwrap_or((LineType::Unknown, None));
let loc = SourceLocation { path, line };
if desc != LineType::TestMain && !temp_map.contains_key(&loc) {
temp_map.insert(
loc,
TracerData {
address: Some(address),
trace_type: desc,
length: 1,
fn_name,
},
);
}
}
}
}
}
}
for (k, v) in &temp_map {
if result.contains_key(k) {
let x = result.get_mut(k).unwrap();
x.push(v.clone());
} else {
result.insert(k.clone(), vec![v.clone()]);
}
}
}
Ok(())
}
fn get_line_addresses(
endian: RunTimeEndian,
project: &Path,
obj: &OFile,
analysis: &HashMap<PathBuf, LineAnalysis>,
config: &Config,
) -> Result<TraceMap> {
let mut result = TraceMap::new();
let debug_info = obj.section_data_by_name(".debug_info").unwrap_or_default();
let debug_info = DebugInfo::new(&debug_info, endian);
let debug_abbrev = obj
.section_data_by_name(".debug_abbrev")
.unwrap_or_default();
let debug_abbrev = DebugAbbrev::new(&debug_abbrev, endian);
let debug_strings = obj.section_data_by_name(".debug_str").unwrap_or_default();
let debug_strings = DebugStr::new(&debug_strings, endian);
let debug_line = obj.section_data_by_name(".debug_line").unwrap_or_default();
let debug_line = DebugLine::new(&debug_line, endian);
let mut iter = debug_info.units();
while let Ok(Some(cu)) = iter.next() {
let addr_size = cu.address_size();
let abbr = match cu.abbreviations(&debug_abbrev) {
Ok(a) => a,
_ => continue,
};
let entry_points = get_entry_points(&cu, &abbr, &debug_strings);
let entries = entry_points
.iter()
.map(|(a, b, c, fn_name)| match c {
FunctionType::Test => (*a, LineType::TestEntry(*b), fn_name),
FunctionType::Standard => (*a, LineType::FunctionEntry(*b), fn_name),
FunctionType::Generated => (*a, LineType::TestMain, fn_name),
})
.collect::<Vec<_>>();
if let Ok(Some((_, root))) = cu.entries(&abbr).next_dfs() {
let offset = match root.attr_value(DW_AT_stmt_list) {
Ok(Some(AttributeValue::DebugLineRef(o))) => o,
_ => continue,
};
let prog = debug_line.program(offset, addr_size, None, None)?;
let mut temp_map: HashMap<SourceLocation, Vec<TracerData>> = HashMap::new();
if let Err(e) =
get_addresses_from_program(prog, &debug_strings, &entries, project, &mut temp_map)
{
debug!("Potential issue reading test addresses {}", e);
} else {
for v in temp_map.values_mut() {
v.dedup_by_key(|x| x.address);
}
let temp_map = temp_map
.into_iter()
.filter(|&(ref k, _)| {
!(config.ignore_tests && k.path.starts_with(project.join("tests")))
})
.filter(|&(ref k, _)| !(config.exclude_path(&k.path)))
.filter(|&(ref k, _)| {
!analysis.should_ignore(k.path.as_ref(), &(k.line as usize))
})
.collect::<HashMap<SourceLocation, Vec<TracerData>>>();
let mut tracemap = TraceMap::new();
for (k, val) in &temp_map {
for v in val.iter() {
let rpath = config.strip_base_dir(&k.path);
match v.address {
Some(ref a) => trace!(
"Adding trace at address 0x{:x} in {}:{}",
a,
rpath.display(),
k.line
),
None => trace!(
"Adding trace with no address at {}:{}",
rpath.display(),
k.line
),
}
tracemap.add_trace(
&k.path,
Trace {
line: k.line,
address: v.address,
length: 1,
stats: CoverageStat::Line(0),
fn_name: v.fn_name.clone(),
},
);
}
}
result.merge(&tracemap);
}
}
}
for (file, ref line_analysis) in analysis.iter() {
if config.exclude_path(file) {
continue;
}
for line in &line_analysis.cover {
let line = *line as u64;
if !result.contains_location(file, line) && !line_analysis.should_ignore(line as usize)
{
let rpath = config.strip_base_dir(file);
trace!(
"Adding trace for potentially uncoverable line in {}:{}",
rpath.display(),
line
);
result.add_trace(
file,
Trace {
line,
address: None,
length: 0,
stats: CoverageStat::Line(0),
fn_name: None,
},
);
}
}
}
Ok(result)
}
#[cfg(target_os = "linux")]
fn open_symbols_file(test: &Path) -> io::Result<File> {
File::open(test)
}
#[cfg(target_os = "macos")]
fn open_symbols_file(test: &Path) -> io::Result<File> {
let d_sym = test.with_extension("dSYM");
File::open(&d_sym)
}
pub fn generate_tracemap(
project: &Workspace,
test: &Path,
analysis: &HashMap<PathBuf, LineAnalysis>,
config: &Config,
) -> io::Result<TraceMap> {
let manifest = project.root();
let file = open_symbols_file(test)?;
let file = unsafe { MmapOptions::new().map(&file)? };
if let Ok(obj) = OFile::parse(&*file) {
let endian = if obj.is_little_endian() {
RunTimeEndian::Little
} else {
RunTimeEndian::Big
};
if let Ok(result) = get_line_addresses(endian, manifest, &obj, &analysis, config) {
Ok(result)
} else {
Err(io::Error::new(
io::ErrorKind::InvalidData,
"Error while parsing",
))
}
} else {
Err(io::Error::new(
io::ErrorKind::InvalidData,
"Unable to parse binary.",
))
}
}