rustc_mir_transform/coverage/
query.rs1use rustc_hir::attrs::CoverageAttrKind;
2use rustc_hir::find_attr;
3use rustc_index::bit_set::DenseBitSet;
4use rustc_middle::middle::codegen_fn_attrs::CodegenFnAttrFlags;
5use rustc_middle::mir::coverage::{BasicCoverageBlock, CoverageIdsInfo, CoverageKind, MappingKind};
6use rustc_middle::mir::{Body, Statement, StatementKind};
7use rustc_middle::ty::{self, TyCtxt};
8use rustc_middle::util::Providers;
9use rustc_span::def_id::LocalDefId;
10use tracing::trace;
11
12use crate::coverage::counters::node_flow::make_node_counters;
13use crate::coverage::counters::{CoverageCounters, transcribe_counters};
14
15pub(crate) fn provide(providers: &mut Providers) {
17 providers.hooks.is_eligible_for_coverage = is_eligible_for_coverage;
18 providers.queries.coverage_attr_on = coverage_attr_on;
19 providers.queries.coverage_ids_info = coverage_ids_info;
20}
21
22fn is_eligible_for_coverage(tcx: TyCtxt<'_>, def_id: LocalDefId) -> bool {
24 if !tcx.def_kind(def_id).is_fn_like() {
32 trace!("InstrumentCoverage skipped for {def_id:?} (not an fn-like)");
33 return false;
34 }
35
36 if tcx.codegen_fn_attrs(def_id).flags.contains(CodegenFnAttrFlags::NAKED) {
37 trace!("InstrumentCoverage skipped for {def_id:?} (`#[naked]`)");
38 return false;
39 }
40
41 if !tcx.coverage_attr_on(def_id) {
42 trace!("InstrumentCoverage skipped for {def_id:?} (`#[coverage(off)]`)");
43 return false;
44 }
45
46 true
47}
48
49fn coverage_attr_on(tcx: TyCtxt<'_>, def_id: LocalDefId) -> bool {
51 if let Some(kind) = find_attr!(tcx, def_id, Coverage(_sp, kind) => kind) {
53 match kind {
54 CoverageAttrKind::On => return true,
55 CoverageAttrKind::Off => return false,
56 }
57 };
58
59 if tcx.is_automatically_derived(def_id.to_def_id()) {
65 return false;
66 }
67
68 match tcx.opt_local_parent(def_id) {
71 Some(parent) => tcx.coverage_attr_on(parent),
72 None => true,
75 }
76}
77
78fn coverage_ids_info<'tcx>(
80 tcx: TyCtxt<'tcx>,
81 instance_def: ty::InstanceKind<'tcx>,
82) -> Option<CoverageIdsInfo> {
83 let mir_body = tcx.instance_mir(instance_def);
84 let fn_cov_info = mir_body.function_coverage_info.as_deref()?;
85
86 let mut bcbs_seen = DenseBitSet::new_empty(fn_cov_info.priority_list.len());
89 for kind in all_coverage_in_mir_body(mir_body) {
90 match *kind {
91 CoverageKind::VirtualCounter { bcb } => {
92 bcbs_seen.insert(bcb);
93 }
94 _ => {}
95 }
96 }
97
98 let mut bcb_needs_counter =
102 DenseBitSet::<BasicCoverageBlock>::new_empty(fn_cov_info.priority_list.len());
103 for mapping in &fn_cov_info.mappings {
104 match mapping.kind {
105 MappingKind::Code { bcb } => {
106 bcb_needs_counter.insert(bcb);
107 }
108 MappingKind::Branch { true_bcb, false_bcb } => {
109 bcb_needs_counter.insert(true_bcb);
110 bcb_needs_counter.insert(false_bcb);
111 }
112 }
113 }
114
115 let mut priority_list = fn_cov_info.priority_list.clone();
117 debug_assert_eq!(priority_list[0], priority_list.iter().copied().max().unwrap());
120 assert!(!bcbs_seen.contains(priority_list[0]));
121 priority_list[1..].sort_by_key(|&bcb| !bcbs_seen.contains(bcb));
127
128 let node_counters = make_node_counters(&fn_cov_info.node_flow_data, &priority_list);
129 let coverage_counters = transcribe_counters(&node_counters, &bcb_needs_counter, &bcbs_seen);
130
131 let CoverageCounters {
132 phys_counter_for_node, next_counter_id, node_counters, expressions, ..
133 } = coverage_counters;
134
135 Some(CoverageIdsInfo {
136 num_counters: next_counter_id.as_u32(),
137 phys_counter_for_node,
138 term_for_bcb: node_counters,
139 expressions,
140 })
141}
142
143fn all_coverage_in_mir_body<'a, 'tcx>(
144 body: &'a Body<'tcx>,
145) -> impl Iterator<Item = &'a CoverageKind> {
146 body.basic_blocks.iter().flat_map(|bb_data| &bb_data.statements).filter_map(|statement| {
147 match statement.kind {
148 StatementKind::Coverage(ref kind) if !is_inlined(body, statement) => Some(kind),
149 _ => None,
150 }
151 })
152}
153
154fn is_inlined(body: &Body<'_>, statement: &Statement<'_>) -> bool {
155 let scope_data = &body.source_scopes[statement.source_info.scope];
156 scope_data.inlined.is_some() || scope_data.inlined_parent_scope.is_some()
157}