coverage: Add -Zcoverage-options
for fine control of coverage
This new nightly-only flag can be used to toggle fine-grained flags that control the details of coverage instrumentation. Currently the only supported flag value is `branch` (or `no-branch`), which is a placeholder for upcoming support for branch coverage. Other flag values can be added in the future, to prototype proposed new behaviour, or to enable special non-default behaviour.
This commit is contained in:
parent
1f544ce305
commit
3407fcc12e
8 changed files with 84 additions and 11 deletions
|
@ -4,11 +4,12 @@ use rustc_data_structures::profiling::TimePassesFormat;
|
||||||
use rustc_errors::{emitter::HumanReadableErrorType, registry, ColorConfig};
|
use rustc_errors::{emitter::HumanReadableErrorType, registry, ColorConfig};
|
||||||
use rustc_session::config::{
|
use rustc_session::config::{
|
||||||
build_configuration, build_session_options, rustc_optgroups, BranchProtection, CFGuard, Cfg,
|
build_configuration, build_session_options, rustc_optgroups, BranchProtection, CFGuard, Cfg,
|
||||||
CollapseMacroDebuginfo, DebugInfo, DumpMonoStatsFormat, ErrorOutputType, ExternEntry,
|
CollapseMacroDebuginfo, CoverageOptions, DebugInfo, DumpMonoStatsFormat, ErrorOutputType,
|
||||||
ExternLocation, Externs, FunctionReturn, InliningThreshold, Input, InstrumentCoverage,
|
ExternEntry, ExternLocation, Externs, FunctionReturn, InliningThreshold, Input,
|
||||||
InstrumentXRay, LinkSelfContained, LinkerPluginLto, LocationDetail, LtoCli, NextSolverConfig,
|
InstrumentCoverage, InstrumentXRay, LinkSelfContained, LinkerPluginLto, LocationDetail, LtoCli,
|
||||||
OomStrategy, Options, OutFileName, OutputType, OutputTypes, PAuthKey, PacRet, Passes, Polonius,
|
NextSolverConfig, OomStrategy, Options, OutFileName, OutputType, OutputTypes, PAuthKey, PacRet,
|
||||||
ProcMacroExecutionStrategy, Strip, SwitchWithOptPath, SymbolManglingVersion, WasiExecModel,
|
Passes, Polonius, ProcMacroExecutionStrategy, Strip, SwitchWithOptPath, SymbolManglingVersion,
|
||||||
|
WasiExecModel,
|
||||||
};
|
};
|
||||||
use rustc_session::lint::Level;
|
use rustc_session::lint::Level;
|
||||||
use rustc_session::search_paths::SearchPath;
|
use rustc_session::search_paths::SearchPath;
|
||||||
|
@ -750,6 +751,7 @@ fn test_unstable_options_tracking_hash() {
|
||||||
);
|
);
|
||||||
tracked!(codegen_backend, Some("abc".to_string()));
|
tracked!(codegen_backend, Some("abc".to_string()));
|
||||||
tracked!(collapse_macro_debuginfo, CollapseMacroDebuginfo::Yes);
|
tracked!(collapse_macro_debuginfo, CollapseMacroDebuginfo::Yes);
|
||||||
|
tracked!(coverage_options, CoverageOptions { branch: true });
|
||||||
tracked!(crate_attr, vec!["abc".to_string()]);
|
tracked!(crate_attr, vec!["abc".to_string()]);
|
||||||
tracked!(cross_crate_inline_threshold, InliningThreshold::Always);
|
tracked!(cross_crate_inline_threshold, InliningThreshold::Always);
|
||||||
tracked!(debug_info_for_profiling, true);
|
tracked!(debug_info_for_profiling, true);
|
||||||
|
|
|
@ -143,6 +143,19 @@ pub enum InstrumentCoverage {
|
||||||
Yes,
|
Yes,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Individual flag values controlled by `-Z coverage-options`.
|
||||||
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
||||||
|
pub struct CoverageOptions {
|
||||||
|
/// Add branch coverage instrumentation (placeholder flag; not yet implemented).
|
||||||
|
pub branch: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for CoverageOptions {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self { branch: false }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Settings for `-Z instrument-xray` flag.
|
/// Settings for `-Z instrument-xray` flag.
|
||||||
#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, Hash)]
|
#[derive(Clone, Copy, Debug, Default, PartialEq, Eq, Hash)]
|
||||||
pub struct InstrumentXRay {
|
pub struct InstrumentXRay {
|
||||||
|
@ -3168,12 +3181,12 @@ pub enum WasiExecModel {
|
||||||
/// how the hash should be calculated when adding a new command-line argument.
|
/// how the hash should be calculated when adding a new command-line argument.
|
||||||
pub(crate) mod dep_tracking {
|
pub(crate) mod dep_tracking {
|
||||||
use super::{
|
use super::{
|
||||||
BranchProtection, CFGuard, CFProtection, CollapseMacroDebuginfo, CrateType, DebugInfo,
|
BranchProtection, CFGuard, CFProtection, CollapseMacroDebuginfo, CoverageOptions,
|
||||||
DebugInfoCompression, ErrorOutputType, FunctionReturn, InliningThreshold,
|
CrateType, DebugInfo, DebugInfoCompression, ErrorOutputType, FunctionReturn,
|
||||||
InstrumentCoverage, InstrumentXRay, LinkerPluginLto, LocationDetail, LtoCli,
|
InliningThreshold, InstrumentCoverage, InstrumentXRay, LinkerPluginLto, LocationDetail,
|
||||||
NextSolverConfig, OomStrategy, OptLevel, OutFileName, OutputType, OutputTypes, Polonius,
|
LtoCli, NextSolverConfig, OomStrategy, OptLevel, OutFileName, OutputType, OutputTypes,
|
||||||
RemapPathScopeComponents, ResolveDocLinks, SourceFileHashAlgorithm, SplitDwarfKind,
|
Polonius, RemapPathScopeComponents, ResolveDocLinks, SourceFileHashAlgorithm,
|
||||||
SwitchWithOptPath, SymbolManglingVersion, WasiExecModel,
|
SplitDwarfKind, SwitchWithOptPath, SymbolManglingVersion, WasiExecModel,
|
||||||
};
|
};
|
||||||
use crate::lint;
|
use crate::lint;
|
||||||
use crate::utils::NativeLib;
|
use crate::utils::NativeLib;
|
||||||
|
@ -3243,6 +3256,7 @@ pub(crate) mod dep_tracking {
|
||||||
CodeModel,
|
CodeModel,
|
||||||
TlsModel,
|
TlsModel,
|
||||||
InstrumentCoverage,
|
InstrumentCoverage,
|
||||||
|
CoverageOptions,
|
||||||
InstrumentXRay,
|
InstrumentXRay,
|
||||||
CrateType,
|
CrateType,
|
||||||
MergeFunctions,
|
MergeFunctions,
|
||||||
|
|
|
@ -396,6 +396,7 @@ mod desc {
|
||||||
pub const parse_optimization_fuel: &str = "crate=integer";
|
pub const parse_optimization_fuel: &str = "crate=integer";
|
||||||
pub const parse_dump_mono_stats: &str = "`markdown` (default) or `json`";
|
pub const parse_dump_mono_stats: &str = "`markdown` (default) or `json`";
|
||||||
pub const parse_instrument_coverage: &str = parse_bool;
|
pub const parse_instrument_coverage: &str = parse_bool;
|
||||||
|
pub const parse_coverage_options: &str = "`branch` or `no-branch`";
|
||||||
pub const parse_instrument_xray: &str = "either a boolean (`yes`, `no`, `on`, `off`, etc), or a comma separated list of settings: `always` or `never` (mutually exclusive), `ignore-loops`, `instruction-threshold=N`, `skip-entry`, `skip-exit`";
|
pub const parse_instrument_xray: &str = "either a boolean (`yes`, `no`, `on`, `off`, etc), or a comma separated list of settings: `always` or `never` (mutually exclusive), `ignore-loops`, `instruction-threshold=N`, `skip-entry`, `skip-exit`";
|
||||||
pub const parse_unpretty: &str = "`string` or `string=string`";
|
pub const parse_unpretty: &str = "`string` or `string=string`";
|
||||||
pub const parse_treat_err_as_bug: &str = "either no value or a non-negative number";
|
pub const parse_treat_err_as_bug: &str = "either no value or a non-negative number";
|
||||||
|
@ -938,6 +939,24 @@ mod parse {
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn parse_coverage_options(slot: &mut CoverageOptions, v: Option<&str>) -> bool {
|
||||||
|
let Some(v) = v else { return true };
|
||||||
|
|
||||||
|
for option in v.split(',') {
|
||||||
|
let (option, enabled) = match option.strip_prefix("no-") {
|
||||||
|
Some(without_no) => (without_no, false),
|
||||||
|
None => (option, true),
|
||||||
|
};
|
||||||
|
let slot = match option {
|
||||||
|
"branch" => &mut slot.branch,
|
||||||
|
_ => return false,
|
||||||
|
};
|
||||||
|
*slot = enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn parse_instrument_xray(
|
pub(crate) fn parse_instrument_xray(
|
||||||
slot: &mut Option<InstrumentXRay>,
|
slot: &mut Option<InstrumentXRay>,
|
||||||
v: Option<&str>,
|
v: Option<&str>,
|
||||||
|
@ -1564,6 +1583,8 @@ options! {
|
||||||
"set option to collapse debuginfo for macros"),
|
"set option to collapse debuginfo for macros"),
|
||||||
combine_cgu: bool = (false, parse_bool, [TRACKED],
|
combine_cgu: bool = (false, parse_bool, [TRACKED],
|
||||||
"combine CGUs into a single one"),
|
"combine CGUs into a single one"),
|
||||||
|
coverage_options: CoverageOptions = (CoverageOptions::default(), parse_coverage_options, [TRACKED],
|
||||||
|
"control details of coverage instrumentation"),
|
||||||
crate_attr: Vec<String> = (Vec::new(), parse_string_push, [TRACKED],
|
crate_attr: Vec<String> = (Vec::new(), parse_string_push, [TRACKED],
|
||||||
"inject the given attribute in the crate"),
|
"inject the given attribute in the crate"),
|
||||||
cross_crate_inline_threshold: InliningThreshold = (InliningThreshold::Sometimes(100), parse_inlining_threshold, [TRACKED],
|
cross_crate_inline_threshold: InliningThreshold = (InliningThreshold::Sometimes(100), parse_inlining_threshold, [TRACKED],
|
||||||
|
|
|
@ -352,6 +352,10 @@ impl Session {
|
||||||
self.opts.cg.instrument_coverage() != InstrumentCoverage::No
|
self.opts.cg.instrument_coverage() != InstrumentCoverage::No
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn instrument_coverage_branch(&self) -> bool {
|
||||||
|
self.instrument_coverage() && self.opts.unstable_opts.coverage_options.branch
|
||||||
|
}
|
||||||
|
|
||||||
pub fn is_sanitizer_cfi_enabled(&self) -> bool {
|
pub fn is_sanitizer_cfi_enabled(&self) -> bool {
|
||||||
self.opts.unstable_opts.sanitizer.contains(SanitizerSet::CFI)
|
self.opts.unstable_opts.sanitizer.contains(SanitizerSet::CFI)
|
||||||
}
|
}
|
||||||
|
|
|
@ -346,6 +346,14 @@ $ llvm-cov report \
|
||||||
more fine-grained coverage options are added.
|
more fine-grained coverage options are added.
|
||||||
Using this value is currently not recommended.
|
Using this value is currently not recommended.
|
||||||
|
|
||||||
|
## `-Z coverage-options=<options>`
|
||||||
|
|
||||||
|
This unstable option provides finer control over some aspects of coverage
|
||||||
|
instrumentation. Pass one or more of the following values, separated by commas.
|
||||||
|
|
||||||
|
- `branch` or `no-branch`
|
||||||
|
- Placeholder for potential branch coverage support in the future.
|
||||||
|
|
||||||
## Other references
|
## Other references
|
||||||
|
|
||||||
Rust's implementation and workflow for source-based code coverage is based on the same library and tools used to implement [source-based code coverage in Clang]. (This document is partially based on the Clang guide.)
|
Rust's implementation and workflow for source-based code coverage is based on the same library and tools used to implement [source-based code coverage in Clang]. (This document is partially based on the Clang guide.)
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
# `coverage-options`
|
||||||
|
|
||||||
|
This option controls details of the coverage instrumentation performed by
|
||||||
|
`-C instrument-coverage`.
|
||||||
|
|
||||||
|
Multiple options can be passed, separated by commas. Valid options are:
|
||||||
|
|
||||||
|
- `branch` or `no-branch`: Placeholder for future branch coverage support.
|
2
tests/ui/instrument-coverage/coverage-options.bad.stderr
Normal file
2
tests/ui/instrument-coverage/coverage-options.bad.stderr
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
error: incorrect value `bad` for unstable option `coverage-options` - `branch` or `no-branch` was expected
|
||||||
|
|
14
tests/ui/instrument-coverage/coverage-options.rs
Normal file
14
tests/ui/instrument-coverage/coverage-options.rs
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
//@ needs-profiler-support
|
||||||
|
//@ revisions: branch no-branch bad
|
||||||
|
//@ compile-flags -Cinstrument-coverage
|
||||||
|
|
||||||
|
//@ [branch] check-pass
|
||||||
|
//@ [branch] compile-flags: -Zcoverage-options=branch
|
||||||
|
|
||||||
|
//@ [no-branch] check-pass
|
||||||
|
//@ [no-branch] compile-flags: -Zcoverage-options=no-branch
|
||||||
|
|
||||||
|
//@ [bad] check-fail
|
||||||
|
//@ [bad] compile-flags: -Zcoverage-options=bad
|
||||||
|
|
||||||
|
fn main() {}
|
Loading…
Add table
Reference in a new issue