Skip to content

Commit 3a6c74b

Browse files
committed
coverage: Separate branch pairs from other mapping kinds
This clears the way for larger changes to how branches are handled by the coverage instrumentor, in order to support branch coverage for more language constructs.
1 parent 999e192 commit 3a6c74b

File tree

3 files changed

+44
-25
lines changed

3 files changed

+44
-25
lines changed

compiler/rustc_mir_transform/src/coverage/mod.rs

+14-8
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ mod tests;
99

1010
use self::counters::{CounterIncrementSite, CoverageCounters};
1111
use self::graph::{BasicCoverageBlock, CoverageGraph};
12-
use self::spans::{BcbMapping, BcbMappingKind, CoverageSpans};
12+
use self::spans::{BcbBranchPair, BcbMapping, BcbMappingKind, CoverageSpans};
1313

1414
use crate::MirPass;
1515

@@ -141,14 +141,10 @@ fn create_mappings<'tcx>(
141141

142142
let mut mappings = Vec::new();
143143

144-
mappings.extend(coverage_spans.all_bcb_mappings().filter_map(
144+
mappings.extend(coverage_spans.mappings.iter().filter_map(
145145
|BcbMapping { kind: bcb_mapping_kind, span }| {
146146
let kind = match *bcb_mapping_kind {
147147
BcbMappingKind::Code(bcb) => MappingKind::Code(term_for_bcb(bcb)),
148-
BcbMappingKind::Branch { true_bcb, false_bcb } => MappingKind::Branch {
149-
true_term: term_for_bcb(true_bcb),
150-
false_term: term_for_bcb(false_bcb),
151-
},
152148
BcbMappingKind::MCDCBranch { true_bcb, false_bcb, condition_info: None } => {
153149
MappingKind::Branch {
154150
true_term: term_for_bcb(true_bcb),
@@ -173,6 +169,16 @@ fn create_mappings<'tcx>(
173169
},
174170
));
175171

172+
mappings.extend(coverage_spans.branch_pairs.iter().filter_map(
173+
|&BcbBranchPair { span, true_bcb, false_bcb }| {
174+
let true_term = term_for_bcb(true_bcb);
175+
let false_term = term_for_bcb(false_bcb);
176+
let kind = MappingKind::Branch { true_term, false_term };
177+
let code_region = make_code_region(source_map, file_name, span, body_span)?;
178+
Some(Mapping { kind, code_region })
179+
},
180+
));
181+
176182
mappings
177183
}
178184

@@ -241,7 +247,7 @@ fn inject_mcdc_statements<'tcx>(
241247

242248
// Inject test vector update first because `inject_statement` always insert new statement at head.
243249
for (end_bcbs, bitmap_idx) in
244-
coverage_spans.all_bcb_mappings().filter_map(|mapping| match &mapping.kind {
250+
coverage_spans.mappings.iter().filter_map(|mapping| match &mapping.kind {
245251
BcbMappingKind::MCDCDecision { end_bcbs, bitmap_idx, .. } => {
246252
Some((end_bcbs, *bitmap_idx))
247253
}
@@ -255,7 +261,7 @@ fn inject_mcdc_statements<'tcx>(
255261
}
256262

257263
for (true_bcb, false_bcb, condition_id) in
258-
coverage_spans.all_bcb_mappings().filter_map(|mapping| match mapping.kind {
264+
coverage_spans.mappings.iter().filter_map(|mapping| match mapping.kind {
259265
BcbMappingKind::MCDCBranch { true_bcb, false_bcb, condition_info } => {
260266
Some((true_bcb, false_bcb, condition_info?.condition_id))
261267
}

compiler/rustc_mir_transform/src/coverage/spans.rs

+25-12
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,10 @@ mod from_mir;
1515
pub(super) enum BcbMappingKind {
1616
/// Associates an ordinary executable code span with its corresponding BCB.
1717
Code(BasicCoverageBlock),
18-
/// Associates a branch span with BCBs for its true and false arms.
19-
Branch { true_bcb: BasicCoverageBlock, false_bcb: BasicCoverageBlock },
18+
19+
// Ordinary branch mappings are stored separately, so they don't have a
20+
// variant in this enum.
21+
//
2022
/// Associates a mcdc branch span with condition info besides fields for normal branch.
2123
MCDCBranch {
2224
true_bcb: BasicCoverageBlock,
@@ -35,9 +37,20 @@ pub(super) struct BcbMapping {
3537
pub(super) span: Span,
3638
}
3739

40+
/// This is separate from [`BcbMappingKind`] to help prepare for larger changes
41+
/// that will be needed for improved branch coverage in the future.
42+
/// (See <https://github.com/rust-lang/rust/pull/124217>.)
43+
#[derive(Debug)]
44+
pub(super) struct BcbBranchPair {
45+
pub(super) span: Span,
46+
pub(super) true_bcb: BasicCoverageBlock,
47+
pub(super) false_bcb: BasicCoverageBlock,
48+
}
49+
3850
pub(super) struct CoverageSpans {
3951
bcb_has_mappings: BitSet<BasicCoverageBlock>,
40-
mappings: Vec<BcbMapping>,
52+
pub(super) mappings: Vec<BcbMapping>,
53+
pub(super) branch_pairs: Vec<BcbBranchPair>,
4154
test_vector_bitmap_bytes: u32,
4255
}
4356

@@ -46,10 +59,6 @@ impl CoverageSpans {
4659
self.bcb_has_mappings.contains(bcb)
4760
}
4861

49-
pub(super) fn all_bcb_mappings(&self) -> impl Iterator<Item = &BcbMapping> {
50-
self.mappings.iter()
51-
}
52-
5362
pub(super) fn test_vector_bitmap_bytes(&self) -> u32 {
5463
self.test_vector_bitmap_bytes
5564
}
@@ -65,6 +74,7 @@ pub(super) fn generate_coverage_spans(
6574
basic_coverage_blocks: &CoverageGraph,
6675
) -> Option<CoverageSpans> {
6776
let mut mappings = vec![];
77+
let mut branch_pairs = vec![];
6878

6979
if hir_info.is_async_fn {
7080
// An async function desugars into a function that returns a future,
@@ -86,7 +96,7 @@ pub(super) fn generate_coverage_spans(
8696
BcbMapping { kind: BcbMappingKind::Code(bcb), span }
8797
}));
8898

89-
mappings.extend(from_mir::extract_branch_mappings(
99+
branch_pairs.extend(from_mir::extract_branch_pairs(
90100
mir_body,
91101
hir_info.body_span,
92102
basic_coverage_blocks,
@@ -99,7 +109,7 @@ pub(super) fn generate_coverage_spans(
99109
));
100110
}
101111

102-
if mappings.is_empty() {
112+
if mappings.is_empty() && branch_pairs.is_empty() {
103113
return None;
104114
}
105115

@@ -112,8 +122,7 @@ pub(super) fn generate_coverage_spans(
112122
for BcbMapping { kind, span: _ } in &mappings {
113123
match *kind {
114124
BcbMappingKind::Code(bcb) => insert(bcb),
115-
BcbMappingKind::Branch { true_bcb, false_bcb }
116-
| BcbMappingKind::MCDCBranch { true_bcb, false_bcb, .. } => {
125+
BcbMappingKind::MCDCBranch { true_bcb, false_bcb, .. } => {
117126
insert(true_bcb);
118127
insert(false_bcb);
119128
}
@@ -126,8 +135,12 @@ pub(super) fn generate_coverage_spans(
126135
}
127136
}
128137
}
138+
for &BcbBranchPair { true_bcb, false_bcb, .. } in &branch_pairs {
139+
insert(true_bcb);
140+
insert(false_bcb);
141+
}
129142

130-
Some(CoverageSpans { bcb_has_mappings, mappings, test_vector_bitmap_bytes })
143+
Some(CoverageSpans { bcb_has_mappings, mappings, branch_pairs, test_vector_bitmap_bytes })
131144
}
132145

133146
#[derive(Debug)]

compiler/rustc_mir_transform/src/coverage/spans/from_mir.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ use rustc_span::{ExpnKind, MacroKind, Span, Symbol};
1313
use crate::coverage::graph::{
1414
BasicCoverageBlock, BasicCoverageBlockData, CoverageGraph, START_BCB,
1515
};
16-
use crate::coverage::spans::{BcbMapping, BcbMappingKind};
16+
use crate::coverage::spans::{BcbBranchPair, BcbMapping, BcbMappingKind};
1717
use crate::coverage::ExtractedHirInfo;
1818

1919
/// Traverses the MIR body to produce an initial collection of coverage-relevant
@@ -388,16 +388,16 @@ fn resolve_block_markers(
388388
}
389389

390390
// FIXME: There is currently a lot of redundancy between
391-
// `extract_branch_mappings` and `extract_mcdc_mappings`. This is needed so
391+
// `extract_branch_pairs` and `extract_mcdc_mappings`. This is needed so
392392
// that they can each be modified without interfering with the other, but in
393393
// the long term we should try to bring them together again when branch coverage
394394
// and MC/DC coverage support are more mature.
395395

396-
pub(super) fn extract_branch_mappings(
396+
pub(super) fn extract_branch_pairs(
397397
mir_body: &mir::Body<'_>,
398398
body_span: Span,
399399
basic_coverage_blocks: &CoverageGraph,
400-
) -> Vec<BcbMapping> {
400+
) -> Vec<BcbBranchPair> {
401401
let Some(branch_info) = mir_body.coverage_branch_info.as_deref() else { return vec![] };
402402

403403
let block_markers = resolve_block_markers(branch_info, mir_body);
@@ -419,7 +419,7 @@ pub(super) fn extract_branch_mappings(
419419
let true_bcb = bcb_from_marker(true_marker)?;
420420
let false_bcb = bcb_from_marker(false_marker)?;
421421

422-
Some(BcbMapping { kind: BcbMappingKind::Branch { true_bcb, false_bcb }, span })
422+
Some(BcbBranchPair { span, true_bcb, false_bcb })
423423
})
424424
.collect::<Vec<_>>()
425425
}

0 commit comments

Comments
 (0)