rustc_mir_transform/coverage/
mod.rs1use rustc_middle::mir::coverage::{CoverageKind, FunctionCoverageInfo};
2use rustc_middle::mir::{self, BasicBlock, Statement, StatementKind, TerminatorKind};
3use rustc_middle::ty::TyCtxt;
4use tracing::{debug, debug_span, trace};
5
6use crate::coverage::counters::BcbCountersData;
7use crate::coverage::graph::CoverageGraph;
8use crate::coverage::mappings::ExtractedMappings;
9
10mod counters;
11mod expansion;
12mod from_mir;
13mod graph;
14mod hir_info;
15mod mappings;
16pub(super) mod query;
17mod spans;
18#[cfg(test)]
19mod tests;
20
21pub(super) struct InstrumentCoverage;
25
26impl<'tcx> crate::MirPass<'tcx> for InstrumentCoverage {
27 fn is_enabled(&self, sess: &rustc_session::Session) -> bool {
28 sess.instrument_coverage()
29 }
30
31 fn run_pass(&self, tcx: TyCtxt<'tcx>, mir_body: &mut mir::Body<'tcx>) {
32 let mir_source = mir_body.source;
33
34 assert!(mir_source.promoted.is_none());
37
38 let def_id = mir_source.def_id().expect_local();
39
40 if !tcx.is_eligible_for_coverage(def_id) {
41 trace!("InstrumentCoverage skipped for {def_id:?} (not eligible)");
42 return;
43 }
44
45 match mir_body.basic_blocks[mir::START_BLOCK].terminator().kind {
48 TerminatorKind::Unreachable => {
49 trace!("InstrumentCoverage skipped for unreachable `START_BLOCK`");
50 return;
51 }
52 _ => {}
53 }
54
55 instrument_function_for_coverage(tcx, mir_body);
56 }
57
58 fn is_required(&self) -> bool {
59 false
60 }
61}
62
63fn instrument_function_for_coverage<'tcx>(tcx: TyCtxt<'tcx>, mir_body: &mut mir::Body<'tcx>) {
64 let def_id = mir_body.source.def_id();
65 let _span = debug_span!("instrument_function_for_coverage", ?def_id).entered();
66
67 let hir_info = hir_info::extract_hir_info(tcx, def_id.expect_local());
68
69 let graph = CoverageGraph::from_mir(mir_body);
72
73 let ExtractedMappings { mappings } =
76 match mappings::extract_mappings_from_mir(tcx, mir_body, &hir_info, &graph) {
77 Ok(m) => m,
78 Err(error) => {
79 tracing::debug!(?error, "mapping extraction failed; skipping this function");
80 return;
81 }
82 };
83
84 let BcbCountersData { node_flow_data, priority_list } =
88 counters::prepare_bcb_counters_data(&graph);
89
90 inject_coverage_statements(mir_body, &graph);
92
93 mir_body.function_coverage_info = Some(Box::new(FunctionCoverageInfo {
94 function_source_hash: hir_info.function_source_hash,
95
96 node_flow_data,
97 priority_list,
98
99 mappings,
100 }));
101}
102
103fn inject_coverage_statements<'tcx>(mir_body: &mut mir::Body<'tcx>, graph: &CoverageGraph) {
105 for (bcb, data) in graph.iter_enumerated() {
106 let target_bb = data.leader_bb();
107 inject_statement(mir_body, CoverageKind::VirtualCounter { bcb }, target_bb);
108 }
109}
110
111fn inject_statement(mir_body: &mut mir::Body<'_>, counter_kind: CoverageKind, bb: BasicBlock) {
112 debug!(" injecting statement {counter_kind:?} for {bb:?}");
113 let data = &mut mir_body[bb];
114 let source_info = data.terminator().source_info;
115 let statement = Statement::new(source_info, StatementKind::Coverage(counter_kind));
116 data.statements.insert(0, statement);
117}