rustc_mir_transform/coverage/
spans.rs1use rustc_middle::mir::coverage::{Mapping, MappingKind, START_BCB};
2use rustc_middle::ty::TyCtxt;
3use rustc_span::source_map::SourceMap;
4use rustc_span::{BytePos, DesugaringKind, ExpnId, ExpnKind, MacroKind, Span};
5use tracing::instrument;
6
7use crate::coverage::expansion::{ExpnTree, SpanWithBcb};
8use crate::coverage::graph::{BasicCoverageBlock, CoverageGraph};
9use crate::coverage::hir_info::ExtractedHirInfo;
10
11pub(super) fn extract_refined_covspans<'tcx>(
12 tcx: TyCtxt<'tcx>,
13 hir_info: &ExtractedHirInfo,
14 graph: &CoverageGraph,
15 expn_tree: &ExpnTree,
16 mappings: &mut Vec<Mapping>,
17) {
18 if hir_info.is_async_fn {
19 if let Some(span) = hir_info.fn_sig_span {
24 mappings.push(Mapping { span, kind: MappingKind::Code { bcb: START_BCB } })
25 }
26 return;
27 }
28
29 let &ExtractedHirInfo { body_span, .. } = hir_info;
30
31 let Some(node) = expn_tree.get(body_span.ctxt().outer_expn()) else { return };
34
35 let mut covspans = vec![];
36
37 for &SpanWithBcb { span, bcb } in &node.spans {
38 covspans.push(Covspan { span, bcb });
39 }
40
41 for &child_expn_id in &node.child_expn_ids {
44 if let Some(covspan) = single_covspan_for_child_expn(tcx, graph, &expn_tree, child_expn_id)
45 {
46 covspans.push(covspan);
47 }
48 }
49
50 covspans.retain(|covspan: &Covspan| {
51 let covspan_span = covspan.span;
52 if !body_span.contains(covspan_span) || body_span.source_equal(covspan_span) {
56 return false;
57 }
58
59 if !body_span.eq_ctxt(covspan_span) {
62 debug_assert!(
63 false,
64 "span context mismatch: body_span={body_span:?}, covspan.span={covspan_span:?}"
65 );
66 return false;
67 }
68
69 true
70 });
71
72 if covspans.is_empty() {
74 return;
75 }
76
77 covspans.push(Covspan {
82 span: hir_info.fn_sig_span.unwrap_or_else(|| body_span.shrink_to_lo()),
83 bcb: START_BCB,
84 });
85
86 let compare_covspans = |a: &Covspan, b: &Covspan| {
87 compare_spans(a.span, b.span)
88 .then_with(|| graph.cmp_in_dominator_order(a.bcb, b.bcb).reverse())
90 };
91 covspans.sort_by(compare_covspans);
92
93 covspans.dedup_by(|b, a| a.span.source_equal(b.span));
98
99 let mut holes = node.hole_spans.iter().copied().map(|span| Hole { span }).collect::<Vec<_>>();
101
102 holes.sort_by(|a, b| compare_spans(a.span, b.span));
103 holes.dedup_by(|b, a| a.merge_if_overlapping_or_adjacent(b));
104
105 discard_spans_overlapping_holes(&mut covspans, &holes);
107
108 let mut covspans = remove_unwanted_overlapping_spans(covspans);
110
111 let source_map = tcx.sess.source_map();
113 covspans.retain_mut(|covspan| {
114 let Some(span) = ensure_non_empty_span(source_map, covspan.span) else { return false };
115 covspan.span = span;
116 true
117 });
118
119 covspans.dedup_by(|b, a| a.merge_if_eligible(b));
121
122 mappings.extend(covspans.into_iter().map(|Covspan { span, bcb }| {
123 Mapping { span, kind: MappingKind::Code { bcb } }
125 }));
126}
127
128fn single_covspan_for_child_expn(
130 tcx: TyCtxt<'_>,
131 graph: &CoverageGraph,
132 expn_tree: &ExpnTree,
133 expn_id: ExpnId,
134) -> Option<Covspan> {
135 let node = expn_tree.get(expn_id)?;
136
137 let bcbs =
138 expn_tree.iter_node_and_descendants(expn_id).flat_map(|n| n.spans.iter().map(|s| s.bcb));
139
140 let bcb = match node.expn_kind {
141 ExpnKind::Macro(MacroKind::Bang, _) | ExpnKind::Desugaring(DesugaringKind::Await) => {
144 bcbs.min_by(|&a, &b| graph.cmp_in_dominator_order(a, b))?
145 }
146 _ => bcbs.max_by(|&a, &b| graph.cmp_in_dominator_order(a, b))?,
149 };
150
151 let mut span = node.call_site?;
154 if matches!(node.expn_kind, ExpnKind::Macro(MacroKind::Bang, _)) {
155 span = tcx.sess.source_map().span_through_char(span, '!');
156 }
157
158 Some(Covspan { span, bcb })
159}
160
161fn discard_spans_overlapping_holes(covspans: &mut Vec<Covspan>, holes: &[Hole]) {
166 debug_assert!(covspans.is_sorted_by(|a, b| compare_spans(a.span, b.span).is_le()));
167 debug_assert!(holes.is_sorted_by(|a, b| compare_spans(a.span, b.span).is_le()));
168 debug_assert!(holes.array_windows().all(|[a, b]| !a.span.overlaps_or_adjacent(b.span)));
169
170 let mut curr_hole = 0usize;
171 let mut overlaps_hole = |covspan: &Covspan| -> bool {
172 while let Some(hole) = holes.get(curr_hole) {
173 if hole.span.hi() <= covspan.span.lo() {
176 curr_hole += 1;
177 continue;
178 }
179
180 return hole.span.overlaps(covspan.span);
181 }
182
183 false
185 };
186
187 covspans.retain(|covspan| !overlaps_hole(covspan));
188}
189
190#[instrument(level = "debug")]
193fn remove_unwanted_overlapping_spans(sorted_spans: Vec<Covspan>) -> Vec<Covspan> {
194 debug_assert!(sorted_spans.is_sorted_by(|a, b| compare_spans(a.span, b.span).is_le()));
195
196 let mut pending = vec![];
199 let mut refined = vec![];
200
201 for curr in sorted_spans {
202 pending.retain(|prev: &Covspan| {
203 if prev.span.hi() <= curr.span.lo() {
204 refined.push(prev.clone());
207 false
208 } else {
209 prev.bcb == curr.bcb
213 }
214 });
215 pending.push(curr);
216 }
217
218 refined.extend(pending);
220 refined
221}
222
223#[derive(Clone, Debug)]
224struct Covspan {
225 span: Span,
226 bcb: BasicCoverageBlock,
227}
228
229impl Covspan {
230 fn merge_if_eligible(&mut self, other: &Self) -> bool {
236 let eligible_for_merge =
237 |a: &Self, b: &Self| (a.bcb == b.bcb) && a.span.overlaps_or_adjacent(b.span);
238
239 if eligible_for_merge(self, other) {
240 self.span = self.span.to(other.span);
241 true
242 } else {
243 false
244 }
245 }
246}
247
248fn compare_spans(a: Span, b: Span) -> std::cmp::Ordering {
250 Ord::cmp(&a.lo(), &b.lo())
252 .then_with(|| Ord::cmp(&a.hi(), &b.hi()).reverse())
258}
259
260fn ensure_non_empty_span(source_map: &SourceMap, span: Span) -> Option<Span> {
261 if !span.is_empty() {
262 return Some(span);
263 }
264
265 source_map
267 .span_to_source(span, |src, start, end| try {
268 if src.as_bytes().get(end).copied() == Some(b'{') {
273 Some(span.with_hi(span.hi() + BytePos(1)))
274 } else if start > 0 && src.as_bytes()[start - 1] == b'}' {
275 Some(span.with_lo(span.lo() - BytePos(1)))
276 } else {
277 None
278 }
279 })
280 .ok()?
281}
282
283#[derive(Debug)]
284struct Hole {
285 span: Span,
286}
287
288impl Hole {
289 fn merge_if_overlapping_or_adjacent(&mut self, other: &mut Self) -> bool {
290 if !self.span.overlaps_or_adjacent(other.span) {
291 return false;
292 }
293
294 self.span = self.span.to(other.span);
295 true
296 }
297}