Skip to main content

rustc_session/
options.rs

1use std::collections::BTreeMap;
2use std::num::{IntErrorKind, NonZero};
3use std::path::PathBuf;
4use std::str;
5
6use rustc_abi::Align;
7use rustc_ast::attr::version::RustcVersion;
8use rustc_data_structures::fx::FxIndexMap;
9use rustc_data_structures::profiling::TimePassesFormat;
10use rustc_data_structures::stable_hash::StableHasher;
11use rustc_errors::{ColorConfig, TerminalUrl};
12use rustc_feature::UnstableFeatures;
13use rustc_hashes::Hash64;
14use rustc_hir::attrs::CollapseMacroDebuginfo;
15use rustc_macros::{BlobDecodable, Encodable};
16use rustc_span::edition::Edition;
17use rustc_span::{RealFileName, RemapPathScopeComponents, SourceFileHashAlgorithm};
18use rustc_target::spec::{
19    CodeModel, FramePointer, LinkerFlavorCli, MergeFunctions, OnBrokenPipe, PanicStrategy,
20    RelocModel, RelroLevel, SanitizerSet, SplitDebuginfo, StackProtector, SymbolVisibility,
21    TargetTuple, TlsModel,
22};
23
24use crate::config::*;
25use crate::search_paths::SearchPath;
26use crate::utils::NativeLib;
27use crate::{EarlyDiagCtxt, Session, lint};
28
29macro_rules! insert {
30    ($opt_name:ident, $opt_expr:expr, $sub_hashes:expr) => {
31        if $sub_hashes
32            .insert(stringify!($opt_name), $opt_expr as &dyn dep_tracking::DepTrackingHash)
33            .is_some()
34        {
35            panic!("duplicate key in CLI DepTrackingHash: {}", stringify!($opt_name))
36        }
37    };
38}
39
40macro_rules! hash_opt {
41    ($opt_name:ident, $opt_expr:expr, $sub_hashes:expr, $_for_crate_hash: ident, [UNTRACKED]) => {{}};
42    ($opt_name:ident, $opt_expr:expr, $sub_hashes:expr, $_for_crate_hash: ident, [TRACKED]) => {{ insert!($opt_name, $opt_expr, $sub_hashes) }};
43    ($opt_name:ident, $opt_expr:expr, $sub_hashes:expr, $for_crate_hash: ident, [TRACKED_NO_CRATE_HASH]) => {{
44        if !$for_crate_hash {
45            insert!($opt_name, $opt_expr, $sub_hashes)
46        }
47    }};
48    ($opt_name:ident, $opt_expr:expr, $sub_hashes:expr, $_for_crate_hash: ident, [SUBSTRUCT]) => {{}};
49}
50
51macro_rules! hash_substruct {
52    ($opt_name:ident, $opt_expr:expr, $error_format:expr, $for_crate_hash:expr, $hasher:expr, [UNTRACKED]) => {{}};
53    ($opt_name:ident, $opt_expr:expr, $error_format:expr, $for_crate_hash:expr, $hasher:expr, [TRACKED]) => {{}};
54    ($opt_name:ident, $opt_expr:expr, $error_format:expr, $for_crate_hash:expr, $hasher:expr, [TRACKED_NO_CRATE_HASH]) => {{}};
55    ($opt_name:ident, $opt_expr:expr, $error_format:expr, $for_crate_hash:expr, $hasher:expr, [SUBSTRUCT]) => {{
56        use crate::config::dep_tracking::DepTrackingHash;
57        $opt_expr.dep_tracking_hash($for_crate_hash, $error_format).hash(
58            $hasher,
59            $error_format,
60            $for_crate_hash,
61        );
62    }};
63}
64
65/// Extended target modifier info.
66/// For example, when external target modifier is '-Zregparm=2':
67/// Target modifier enum value + user value ('2') from external crate
68/// is converted into description: prefix ('Z'), name ('regparm'), tech value ('Some(2)').
69pub struct ExtendedTargetModifierInfo {
70    /// Flag prefix (usually, 'C' for codegen flags or 'Z' for unstable flags)
71    pub prefix: String,
72    /// Flag name
73    pub name: String,
74    /// Flag parsed technical value
75    pub tech_value: String,
76}
77
78/// A recorded -Zopt_name=opt_value (or -Copt_name=opt_value)
79/// which alter the ABI or effectiveness of exploit mitigations.
80#[derive(#[automatically_derived]
impl ::core::fmt::Debug for TargetModifier {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f,
            "TargetModifier", "opt", &self.opt, "value_name",
            &&self.value_name)
    }
}Debug, #[automatically_derived]
impl ::core::clone::Clone for TargetModifier {
    #[inline]
    fn clone(&self) -> TargetModifier {
        TargetModifier {
            opt: ::core::clone::Clone::clone(&self.opt),
            value_name: ::core::clone::Clone::clone(&self.value_name),
        }
    }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for TargetModifier {
    #[inline]
    fn eq(&self, other: &TargetModifier) -> bool {
        self.opt == other.opt && self.value_name == other.value_name
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for TargetModifier {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<OptionsTargetModifiers>;
        let _: ::core::cmp::AssertParamIsEq<String>;
    }
}Eq, #[automatically_derived]
impl ::core::cmp::PartialOrd for TargetModifier {
    #[inline]
    fn partial_cmp(&self, other: &TargetModifier)
        -> ::core::option::Option<::core::cmp::Ordering> {
        ::core::option::Option::Some(::core::cmp::Ord::cmp(self, other))
    }
}PartialOrd, #[automatically_derived]
impl ::core::cmp::Ord for TargetModifier {
    #[inline]
    fn cmp(&self, other: &TargetModifier) -> ::core::cmp::Ordering {
        match ::core::cmp::Ord::cmp(&self.opt, &other.opt) {
            ::core::cmp::Ordering::Equal =>
                ::core::cmp::Ord::cmp(&self.value_name, &other.value_name),
            cmp => cmp,
        }
    }
}Ord, const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for TargetModifier {
            fn encode(&self, __encoder: &mut __E) {
                match *self {
                    TargetModifier {
                        opt: ref __binding_0, value_name: ref __binding_1 } => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_1,
                            __encoder);
                    }
                }
            }
        }
    };Encodable, const _: () =
    {
        impl<__D: ::rustc_span::BlobDecoder> ::rustc_serialize::Decodable<__D>
            for TargetModifier {
            fn decode(__decoder: &mut __D) -> Self {
                TargetModifier {
                    opt: ::rustc_serialize::Decodable::decode(__decoder),
                    value_name: ::rustc_serialize::Decodable::decode(__decoder),
                }
            }
        }
    };BlobDecodable)]
81pub struct TargetModifier {
82    /// Option enum value
83    pub opt: OptionsTargetModifiers,
84    /// User-provided option value (before parsing)
85    pub value_name: String,
86}
87
88pub mod mitigation_coverage;
89
90mod target_modifier_consistency_check {
91    use super::*;
92    pub(super) fn sanitizer(l: &TargetModifier, r: Option<&TargetModifier>) -> bool {
93        let mut lparsed: SanitizerSet = Default::default();
94        let lval = if l.value_name.is_empty() { None } else { Some(l.value_name.as_str()) };
95        parse::parse_sanitizers(&mut lparsed, lval);
96
97        let mut rparsed: SanitizerSet = Default::default();
98        let rval = r.filter(|v| !v.value_name.is_empty()).map(|v| v.value_name.as_str());
99        parse::parse_sanitizers(&mut rparsed, rval);
100
101        // Some sanitizers need to be target modifiers, and some do not.
102        // For now, we should mark all sanitizers as target modifiers except for these:
103        // AddressSanitizer, LeakSanitizer
104        let tmod_sanitizers = SanitizerSet::MEMORY
105            | SanitizerSet::THREAD
106            | SanitizerSet::HWADDRESS
107            | SanitizerSet::CFI
108            | SanitizerSet::MEMTAG
109            | SanitizerSet::SHADOWCALLSTACK
110            | SanitizerSet::KCFI
111            | SanitizerSet::KERNELADDRESS
112            | SanitizerSet::KERNELHWADDRESS
113            | SanitizerSet::SAFESTACK
114            | SanitizerSet::DATAFLOW;
115
116        lparsed & tmod_sanitizers == rparsed & tmod_sanitizers
117    }
118    pub(super) fn sanitizer_cfi_normalize_integers(
119        sess: &Session,
120        l: &TargetModifier,
121        r: Option<&TargetModifier>,
122    ) -> bool {
123        // For kCFI, the helper flag -Zsanitizer-cfi-normalize-integers should also be a target modifier
124        if sess.sanitizers().contains(SanitizerSet::KCFI) {
125            if let Some(r) = r {
126                return l.extend().tech_value == r.extend().tech_value;
127            } else {
128                return false;
129            }
130        }
131        true
132    }
133}
134
135impl TargetModifier {
136    pub fn extend(&self) -> ExtendedTargetModifierInfo {
137        self.opt.reparse(&self.value_name)
138    }
139    // Custom consistency check for target modifiers (or default `l.tech_value == r.tech_value`)
140    // When other is None, consistency with default value is checked
141    pub fn consistent(&self, sess: &Session, other: Option<&TargetModifier>) -> bool {
142        if !(other.is_none() || self.opt == other.unwrap().opt) {
    ::core::panicking::panic("assertion failed: other.is_none() || self.opt == other.unwrap().opt")
};assert!(other.is_none() || self.opt == other.unwrap().opt);
143        match self.opt {
144            OptionsTargetModifiers::UnstableOptions(unstable) => match unstable {
145                UnstableOptionsTargetModifiers::Sanitizer => {
146                    return target_modifier_consistency_check::sanitizer(self, other);
147                }
148                UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers => {
149                    return target_modifier_consistency_check::sanitizer_cfi_normalize_integers(
150                        sess, self, other,
151                    );
152                }
153                _ => {}
154            },
155            _ => {}
156        };
157        match other {
158            Some(other) => self.extend().tech_value == other.extend().tech_value,
159            None => false,
160        }
161    }
162}
163
164fn tmod_push_impl(
165    opt: OptionsTargetModifiers,
166    tmod_vals: &BTreeMap<OptionsTargetModifiers, String>,
167    tmods: &mut Vec<TargetModifier>,
168) {
169    if let Some(v) = tmod_vals.get(&opt) {
170        tmods.push(TargetModifier { opt, value_name: v.clone() })
171    }
172}
173
174macro_rules! top_level_options {
175    (
176        $(#[$top_level_attr:meta])*
177        pub struct Options {
178            $(
179                $(#[$attr:meta])*
180                $opt:ident : $t:ty
181                [$dep_tracking_marker:ident]
182                $( { TARGET_MODIFIER: $tmod_variant:ident($tmod_enum:ident) } )?
183                ,
184            )*
185        }
186    ) => {
187        #[derive(PartialEq, Eq, PartialOrd, Ord, Debug, Copy, Clone, Encodable, BlobDecodable)]
188        pub enum OptionsTargetModifiers {
189            $(
190                $(
191                    $tmod_variant($tmod_enum),
192                )?
193            )*
194        }
195
196        impl OptionsTargetModifiers {
197            pub fn reparse(&self, user_value: &str) -> ExtendedTargetModifierInfo {
198                match self {
199                    $(
200                        $(
201                            Self::$tmod_variant(v) => v.reparse(user_value),
202                        )?
203                    )*
204                    #[allow(unreachable_patterns)]
205                    _ => panic!("unknown target modifier option: {self:?}"),
206                }
207            }
208
209            pub fn is_target_modifier(flag_name: &str) -> bool {
210                $(
211                    $(
212                        if $tmod_enum::is_target_modifier(flag_name) {
213                            return true
214                        }
215                    )?
216                )*
217                false
218            }
219        }
220
221        #[derive(Clone)]
222        $(#[$top_level_attr])*
223        pub struct Options {
224            $(
225                $(#[$attr])*
226                pub $opt: $t,
227            )*
228            pub target_modifiers: BTreeMap<OptionsTargetModifiers, String>,
229            pub mitigation_coverage_map: mitigation_coverage::MitigationCoverageMap,
230        }
231
232        impl Options {
233            pub fn dep_tracking_hash(&self, for_crate_hash: bool) -> Hash64 {
234                let mut sub_hashes = BTreeMap::new();
235                $(
236                    hash_opt!(
237                        $opt,
238                        &self.$opt,
239                        &mut sub_hashes,
240                        for_crate_hash,
241                        [$dep_tracking_marker]
242                    );
243                )*
244                let mut hasher = StableHasher::new();
245                dep_tracking::stable_hash(
246                    sub_hashes,
247                    &mut hasher,
248                    self.error_format,
249                    for_crate_hash,
250                );
251                $(
252                    hash_substruct!(
253                        $opt,
254                        &self.$opt,
255                        self.error_format,
256                        for_crate_hash,
257                        &mut hasher,
258                        [$dep_tracking_marker]
259                    );
260                )*
261                hasher.finish()
262            }
263
264            pub fn gather_target_modifiers(&self) -> Vec<TargetModifier> {
265                let mut mods = Vec::<TargetModifier>::new();
266                $(
267                    $(
268                        // Only expand for flags that have `TARGET_MODIFIER`.
269                        ${ignore($tmod_enum)}
270                        self.$opt.gather_target_modifiers(&mut mods, &self.target_modifiers);
271                    )?
272                )*
273                mods.sort_by(|a, b| a.opt.cmp(&b.opt));
274                mods
275            }
276        }
277    }
278}
279
280#[automatically_derived]
impl ::core::cmp::PartialEq for OptionsTargetModifiers {
    #[inline]
    fn eq(&self, other: &OptionsTargetModifiers) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr &&
            match (self, other) {
                (OptionsTargetModifiers::UnstableOptions(__self_0),
                    OptionsTargetModifiers::UnstableOptions(__arg1_0)) =>
                    __self_0 == __arg1_0,
                (OptionsTargetModifiers::CodegenOptions(__self_0),
                    OptionsTargetModifiers::CodegenOptions(__arg1_0)) =>
                    __self_0 == __arg1_0,
                _ => unsafe { ::core::intrinsics::unreachable() }
            }
    }
}
#[automatically_derived]
impl ::core::cmp::Eq for OptionsTargetModifiers {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<UnstableOptionsTargetModifiers>;
        let _: ::core::cmp::AssertParamIsEq<CodegenOptionsTargetModifiers>;
    }
}
#[automatically_derived]
impl ::core::cmp::PartialOrd for OptionsTargetModifiers {
    #[inline]
    fn partial_cmp(&self, other: &OptionsTargetModifiers)
        -> ::core::option::Option<::core::cmp::Ordering> {
        ::core::option::Option::Some(::core::cmp::Ord::cmp(self, other))
    }
}
#[automatically_derived]
impl ::core::cmp::Ord for OptionsTargetModifiers {
    #[inline]
    fn cmp(&self, other: &OptionsTargetModifiers) -> ::core::cmp::Ordering {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        match ::core::cmp::Ord::cmp(&__self_discr, &__arg1_discr) {
            ::core::cmp::Ordering::Equal =>
                match (self, other) {
                    (OptionsTargetModifiers::UnstableOptions(__self_0),
                        OptionsTargetModifiers::UnstableOptions(__arg1_0)) =>
                        ::core::cmp::Ord::cmp(__self_0, __arg1_0),
                    (OptionsTargetModifiers::CodegenOptions(__self_0),
                        OptionsTargetModifiers::CodegenOptions(__arg1_0)) =>
                        ::core::cmp::Ord::cmp(__self_0, __arg1_0),
                    _ => unsafe { ::core::intrinsics::unreachable() }
                },
            cmp => cmp,
        }
    }
}
#[automatically_derived]
impl ::core::fmt::Debug for OptionsTargetModifiers {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            OptionsTargetModifiers::UnstableOptions(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "UnstableOptions", &__self_0),
            OptionsTargetModifiers::CodegenOptions(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f,
                    "CodegenOptions", &__self_0),
        }
    }
}
#[automatically_derived]
impl ::core::marker::Copy for OptionsTargetModifiers { }
#[automatically_derived]
#[doc(hidden)]
unsafe impl ::core::clone::TrivialClone for OptionsTargetModifiers { }
#[automatically_derived]
impl ::core::clone::Clone for OptionsTargetModifiers {
    #[inline]
    fn clone(&self) -> OptionsTargetModifiers {
        let _:
                ::core::clone::AssertParamIsClone<UnstableOptionsTargetModifiers>;
        let _:
                ::core::clone::AssertParamIsClone<CodegenOptionsTargetModifiers>;
        *self
    }
}
const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for OptionsTargetModifiers {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        OptionsTargetModifiers::UnstableOptions(ref __binding_0) =>
                            {
                            0usize
                        }
                        OptionsTargetModifiers::CodegenOptions(ref __binding_0) => {
                            1usize
                        }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    OptionsTargetModifiers::UnstableOptions(ref __binding_0) =>
                        {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                    OptionsTargetModifiers::CodegenOptions(ref __binding_0) => {
                        ::rustc_serialize::Encodable::<__E>::encode(__binding_0,
                            __encoder);
                    }
                }
            }
        }
    };
const _: () =
    {
        impl<__D: ::rustc_span::BlobDecoder> ::rustc_serialize::Decodable<__D>
            for OptionsTargetModifiers {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        OptionsTargetModifiers::UnstableOptions(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    1usize => {
                        OptionsTargetModifiers::CodegenOptions(::rustc_serialize::Decodable::decode(__decoder))
                    }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `OptionsTargetModifiers`, expected 0..2, actual {0}",
                                n));
                    }
                }
            }
        }
    };
impl OptionsTargetModifiers {
    pub fn reparse(&self, user_value: &str) -> ExtendedTargetModifierInfo {
        match self {
            Self::UnstableOptions(v) => v.reparse(user_value),
            Self::CodegenOptions(v) =>
                v.reparse(user_value),
                #[allow(unreachable_patterns)]
                _ => {
                ::core::panicking::panic_fmt(format_args!("unknown target modifier option: {0:?}",
                        self));
            }
        }
    }
    pub fn is_target_modifier(flag_name: &str) -> bool {
        if UnstableOptionsTargetModifiers::is_target_modifier(flag_name) {
            return true
        }
        if CodegenOptionsTargetModifiers::is_target_modifier(flag_name) {
            return true
        }
        false
    }
}
#[doc = r" The top-level command-line options struct."]
#[doc = r""]
#[doc =
r" For each option, one has to specify how it behaves with regard to the"]
#[doc =
r" dependency tracking system of incremental compilation. This is done via the"]
#[doc = r" square-bracketed directive after the field type. The options are:"]
#[doc = r""]
#[doc = r" - `[TRACKED]`"]
#[doc =
r" A change in the given field will cause the compiler to completely clear the"]
#[doc = r" incremental compilation cache before proceeding."]
#[doc = r""]
#[doc = r" - `[TRACKED_NO_CRATE_HASH]`"]
#[doc =
r" Same as `[TRACKED]`, but will not affect the crate hash. This is useful for options that"]
#[doc = r" only affect the incremental cache."]
#[doc = r""]
#[doc = r" - `[UNTRACKED]`"]
#[doc = r" Incremental compilation is not influenced by this option."]
#[doc = r""]
#[doc = r" - `[SUBSTRUCT]`"]
#[doc = r" Second-level sub-structs containing more options."]
#[doc = r""]
#[doc =
r" If you add a new option to this struct or one of the sub-structs like"]
#[doc =
r" `CodegenOptions`, think about how it influences incremental compilation. If in"]
#[doc =
r#" doubt, specify `[TRACKED]`, which is always "correct" but might lead to"#]
#[doc = r" unnecessary re-compilation."]
#[rustc_lint_opt_ty]
pub struct Options {
    #[doc =
    r" The crate config requested for the session, which may be combined"]
    #[doc =
    r" with additional crate configurations during the compile process."]
    #[rustc_lint_opt_deny_field_access("use `TyCtxt::crate_types` instead of this field")]
    pub crate_types: Vec<CrateType>,
    pub optimize: OptLevel,
    #[doc =
    r" Include the `debug_assertions` flag in dependency tracking, since it"]
    #[doc = r" can influence whether overflow checks are done or not."]
    pub debug_assertions: bool,
    pub debuginfo: DebugInfo,
    pub lint_opts: Vec<(String, lint::Level)>,
    pub lint_cap: Option<lint::Level>,
    pub describe_lints: bool,
    pub output_types: OutputTypes,
    pub search_paths: Vec<SearchPath>,
    pub libs: Vec<NativeLib>,
    pub sysroot: Sysroot,
    pub target_triple: TargetTuple,
    #[doc =
    r" Effective logical environment used by `env!`/`option_env!` macros"]
    pub logical_env: FxIndexMap<String, String>,
    pub test: bool,
    pub error_format: ErrorOutputType,
    pub diagnostic_width: Option<usize>,
    #[doc = r" If `Some`, enable incremental compilation, using the given"]
    #[doc = r" directory to store intermediate results."]
    pub incremental: Option<PathBuf>,
    pub unstable_opts: UnstableOptions,
    pub prints: Vec<PrintRequest>,
    pub cg: CodegenOptions,
    pub externs: Externs,
    pub crate_name: Option<String>,
    #[doc = r" Indicates how the compiler should treat unstable features."]
    pub unstable_features: UnstableFeatures,
    #[doc =
    r" Indicates whether this run of the compiler is actually rustdoc. This"]
    #[doc =
    r" is currently just a hack and will be removed eventually, so please"]
    #[doc = r" try to not rely on this too much."]
    pub actually_rustdoc: bool,
    #[doc = r" Whether name resolver should resolve documentation links."]
    pub resolve_doc_links: ResolveDocLinks,
    #[doc = r" Control path trimming."]
    pub trimmed_def_paths: bool,
    #[doc =
    r" Specifications of codegen units / ThinLTO which are forced as a"]
    #[doc =
    r" result of parsing command line options. These are not necessarily"]
    #[doc = r" what rustc was invoked with, but massaged a bit to agree with"]
    #[doc =
    r" commands like `--emit llvm-ir` which they're often incompatible with"]
    #[doc = r" if we otherwise use the defaults of rustc."]
    #[rustc_lint_opt_deny_field_access("use `Session::codegen_units` instead of this field")]
    pub cli_forced_codegen_units: Option<usize>,
    #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
    pub cli_forced_local_thinlto_off: bool,
    #[doc =
    r" Remap source path prefixes in all output (messages, object files, debug, etc.)."]
    pub remap_path_prefix: Vec<(PathBuf, PathBuf)>,
    #[doc =
    r" Defines which scopes of paths should be remapped by `--remap-path-prefix`."]
    pub remap_path_scope: RemapPathScopeComponents,
    #[doc =
    r" Base directory containing the `library/` directory for the Rust standard library."]
    #[doc = r" Right now it's always `$sysroot/lib/rustlib/src/rust`"]
    #[doc = r" (i.e. the `rustup` `rust-src` component)."]
    #[doc = r""]
    #[doc =
    r" This directory is what the virtual `/rustc/$hash` is translated back to,"]
    #[doc =
    r" if Rust was built with path remapping to `/rustc/$hash` enabled"]
    #[doc = r" (the `rust.remap-debuginfo` option in `bootstrap.toml`)."]
    pub real_rust_source_base_dir: Option<PathBuf>,
    #[doc =
    r" Base directory containing the `compiler/` directory for the rustc sources."]
    #[doc = r" Right now it's always `$sysroot/lib/rustlib/rustc-src/rust`"]
    #[doc = r" (i.e. the `rustup` `rustc-dev` component)."]
    #[doc = r""]
    #[doc =
    r" This directory is what the virtual `/rustc-dev/$hash` is translated back to,"]
    #[doc =
    r" if Rust was built with path remapping to `/rustc/$hash` enabled"]
    #[doc = r" (the `rust.remap-debuginfo` option in `bootstrap.toml`)."]
    pub real_rustc_dev_source_base_dir: Option<PathBuf>,
    pub edition: Edition,
    #[doc =
    r" `true` if we're emitting JSON blobs about each artifact produced"]
    #[doc = r" by the compiler."]
    pub json_artifact_notifications: bool,
    #[doc =
    r" `true` if we're emitting JSON timings with the start and end of"]
    #[doc = r" high-level compilation sections"]
    pub json_timings: bool,
    #[doc =
    r" `true` if we're emitting a JSON blob containing the unused externs"]
    pub json_unused_externs: JsonUnusedExterns,
    #[doc =
    r" `true` if we're emitting a JSON job containing a future-incompat report for lints"]
    pub json_future_incompat: bool,
    pub pretty: Option<PpMode>,
    #[doc = r" The (potentially remapped) working directory"]
    #[rustc_lint_opt_deny_field_access("use `SourceMap::working_dir` instead of this field")]
    pub working_dir: RealFileName,
    pub color: ColorConfig,
    pub verbose: bool,
    pub target_modifiers: BTreeMap<OptionsTargetModifiers, String>,
    pub mitigation_coverage_map: mitigation_coverage::MitigationCoverageMap,
}
#[automatically_derived]
impl ::core::clone::Clone for Options {
    #[inline]
    fn clone(&self) -> Options {
        Options {
            crate_types: ::core::clone::Clone::clone(&self.crate_types),
            optimize: ::core::clone::Clone::clone(&self.optimize),
            debug_assertions: ::core::clone::Clone::clone(&self.debug_assertions),
            debuginfo: ::core::clone::Clone::clone(&self.debuginfo),
            lint_opts: ::core::clone::Clone::clone(&self.lint_opts),
            lint_cap: ::core::clone::Clone::clone(&self.lint_cap),
            describe_lints: ::core::clone::Clone::clone(&self.describe_lints),
            output_types: ::core::clone::Clone::clone(&self.output_types),
            search_paths: ::core::clone::Clone::clone(&self.search_paths),
            libs: ::core::clone::Clone::clone(&self.libs),
            sysroot: ::core::clone::Clone::clone(&self.sysroot),
            target_triple: ::core::clone::Clone::clone(&self.target_triple),
            logical_env: ::core::clone::Clone::clone(&self.logical_env),
            test: ::core::clone::Clone::clone(&self.test),
            error_format: ::core::clone::Clone::clone(&self.error_format),
            diagnostic_width: ::core::clone::Clone::clone(&self.diagnostic_width),
            incremental: ::core::clone::Clone::clone(&self.incremental),
            unstable_opts: ::core::clone::Clone::clone(&self.unstable_opts),
            prints: ::core::clone::Clone::clone(&self.prints),
            cg: ::core::clone::Clone::clone(&self.cg),
            externs: ::core::clone::Clone::clone(&self.externs),
            crate_name: ::core::clone::Clone::clone(&self.crate_name),
            unstable_features: ::core::clone::Clone::clone(&self.unstable_features),
            actually_rustdoc: ::core::clone::Clone::clone(&self.actually_rustdoc),
            resolve_doc_links: ::core::clone::Clone::clone(&self.resolve_doc_links),
            trimmed_def_paths: ::core::clone::Clone::clone(&self.trimmed_def_paths),
            cli_forced_codegen_units: ::core::clone::Clone::clone(&self.cli_forced_codegen_units),
            cli_forced_local_thinlto_off: ::core::clone::Clone::clone(&self.cli_forced_local_thinlto_off),
            remap_path_prefix: ::core::clone::Clone::clone(&self.remap_path_prefix),
            remap_path_scope: ::core::clone::Clone::clone(&self.remap_path_scope),
            real_rust_source_base_dir: ::core::clone::Clone::clone(&self.real_rust_source_base_dir),
            real_rustc_dev_source_base_dir: ::core::clone::Clone::clone(&self.real_rustc_dev_source_base_dir),
            edition: ::core::clone::Clone::clone(&self.edition),
            json_artifact_notifications: ::core::clone::Clone::clone(&self.json_artifact_notifications),
            json_timings: ::core::clone::Clone::clone(&self.json_timings),
            json_unused_externs: ::core::clone::Clone::clone(&self.json_unused_externs),
            json_future_incompat: ::core::clone::Clone::clone(&self.json_future_incompat),
            pretty: ::core::clone::Clone::clone(&self.pretty),
            working_dir: ::core::clone::Clone::clone(&self.working_dir),
            color: ::core::clone::Clone::clone(&self.color),
            verbose: ::core::clone::Clone::clone(&self.verbose),
            target_modifiers: ::core::clone::Clone::clone(&self.target_modifiers),
            mitigation_coverage_map: ::core::clone::Clone::clone(&self.mitigation_coverage_map),
        }
    }
}
impl Options {
    pub fn dep_tracking_hash(&self, for_crate_hash: bool) -> Hash64 {
        let mut sub_hashes = BTreeMap::new();
        {
            if (&mut sub_hashes).insert("crate_types",
                        &self.crate_types as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "crate_types"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("optimize",
                        &self.optimize as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "optimize"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debug_assertions",
                        &self.debug_assertions as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debug_assertions"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debuginfo",
                        &self.debuginfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debuginfo"));
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("lint_opts",
                            &self.lint_opts as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "lint_opts"));
                    }
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("lint_cap",
                            &self.lint_cap as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "lint_cap"));
                    }
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("output_types",
                        &self.output_types as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "output_types"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("libs",
                        &self.libs as &dyn dep_tracking::DepTrackingHash).is_some()
                {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "libs"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("target_triple",
                        &self.target_triple as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "target_triple"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("logical_env",
                        &self.logical_env as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "logical_env"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("test",
                        &self.test as &dyn dep_tracking::DepTrackingHash).is_some()
                {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "test"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("crate_name",
                        &self.crate_name as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "crate_name"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("unstable_features",
                        &self.unstable_features as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "unstable_features"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("actually_rustdoc",
                        &self.actually_rustdoc as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "actually_rustdoc"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("resolve_doc_links",
                        &self.resolve_doc_links as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "resolve_doc_links"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("trimmed_def_paths",
                        &self.trimmed_def_paths as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "trimmed_def_paths"));
                }
            }
        };
        {};
        {};
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("remap_path_prefix",
                            &self.remap_path_prefix as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "remap_path_prefix"));
                    }
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("remap_path_scope",
                            &self.remap_path_scope as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "remap_path_scope"));
                    }
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("real_rust_source_base_dir",
                            &self.real_rust_source_base_dir as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "real_rust_source_base_dir"));
                    }
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("real_rustc_dev_source_base_dir",
                            &self.real_rustc_dev_source_base_dir as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "real_rustc_dev_source_base_dir"));
                    }
                }
            }
        };
        {
            if (&mut sub_hashes).insert("edition",
                        &self.edition as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "edition"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("json_artifact_notifications",
                        &self.json_artifact_notifications as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "json_artifact_notifications"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("json_future_incompat",
                        &self.json_future_incompat as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "json_future_incompat"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("working_dir",
                        &self.working_dir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "working_dir"));
                }
            }
        };
        {};
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("verbose",
                            &self.verbose as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "verbose"));
                    }
                }
            }
        };
        let mut hasher = StableHasher::new();
        dep_tracking::stable_hash(sub_hashes, &mut hasher, self.error_format,
            for_crate_hash);
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {
            use crate::config::dep_tracking::DepTrackingHash;
            (&self.unstable_opts).dep_tracking_hash(for_crate_hash,
                    self.error_format).hash(&mut hasher, self.error_format,
                for_crate_hash);
        };
        {};
        {
            use crate::config::dep_tracking::DepTrackingHash;
            (&self.cg).dep_tracking_hash(for_crate_hash,
                    self.error_format).hash(&mut hasher, self.error_format,
                for_crate_hash);
        };
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        hasher.finish()
    }
    pub fn gather_target_modifiers(&self) -> Vec<TargetModifier> {
        let mut mods = Vec::<TargetModifier>::new();
        self.unstable_opts.gather_target_modifiers(&mut mods,
            &self.target_modifiers);
        self.cg.gather_target_modifiers(&mut mods, &self.target_modifiers);
        mods.sort_by(|a, b| a.opt.cmp(&b.opt));
        mods
    }
}top_level_options!(
281    /// The top-level command-line options struct.
282    ///
283    /// For each option, one has to specify how it behaves with regard to the
284    /// dependency tracking system of incremental compilation. This is done via the
285    /// square-bracketed directive after the field type. The options are:
286    ///
287    /// - `[TRACKED]`
288    /// A change in the given field will cause the compiler to completely clear the
289    /// incremental compilation cache before proceeding.
290    ///
291    /// - `[TRACKED_NO_CRATE_HASH]`
292    /// Same as `[TRACKED]`, but will not affect the crate hash. This is useful for options that
293    /// only affect the incremental cache.
294    ///
295    /// - `[UNTRACKED]`
296    /// Incremental compilation is not influenced by this option.
297    ///
298    /// - `[SUBSTRUCT]`
299    /// Second-level sub-structs containing more options.
300    ///
301    /// If you add a new option to this struct or one of the sub-structs like
302    /// `CodegenOptions`, think about how it influences incremental compilation. If in
303    /// doubt, specify `[TRACKED]`, which is always "correct" but might lead to
304    /// unnecessary re-compilation.
305    #[rustc_lint_opt_ty]
306    pub struct Options {
307        /// The crate config requested for the session, which may be combined
308        /// with additional crate configurations during the compile process.
309        #[rustc_lint_opt_deny_field_access("use `TyCtxt::crate_types` instead of this field")]
310        crate_types: Vec<CrateType> [TRACKED],
311        optimize: OptLevel [TRACKED],
312        /// Include the `debug_assertions` flag in dependency tracking, since it
313        /// can influence whether overflow checks are done or not.
314        debug_assertions: bool [TRACKED],
315        debuginfo: DebugInfo [TRACKED],
316        lint_opts: Vec<(String, lint::Level)> [TRACKED_NO_CRATE_HASH],
317        lint_cap: Option<lint::Level> [TRACKED_NO_CRATE_HASH],
318        describe_lints: bool [UNTRACKED],
319        output_types: OutputTypes [TRACKED],
320        search_paths: Vec<SearchPath> [UNTRACKED],
321        libs: Vec<NativeLib> [TRACKED],
322        sysroot: Sysroot [UNTRACKED],
323
324        target_triple: TargetTuple [TRACKED],
325
326        /// Effective logical environment used by `env!`/`option_env!` macros
327        logical_env: FxIndexMap<String, String> [TRACKED],
328
329        test: bool [TRACKED],
330        error_format: ErrorOutputType [UNTRACKED],
331        diagnostic_width: Option<usize> [UNTRACKED],
332
333        /// If `Some`, enable incremental compilation, using the given
334        /// directory to store intermediate results.
335        incremental: Option<PathBuf> [UNTRACKED],
336
337        unstable_opts: UnstableOptions [SUBSTRUCT] { TARGET_MODIFIER: UnstableOptions(UnstableOptionsTargetModifiers) },
338        prints: Vec<PrintRequest> [UNTRACKED],
339        cg: CodegenOptions [SUBSTRUCT] { TARGET_MODIFIER: CodegenOptions(CodegenOptionsTargetModifiers) },
340        externs: Externs [UNTRACKED],
341        crate_name: Option<String> [TRACKED],
342        /// Indicates how the compiler should treat unstable features.
343        unstable_features: UnstableFeatures [TRACKED],
344
345        /// Indicates whether this run of the compiler is actually rustdoc. This
346        /// is currently just a hack and will be removed eventually, so please
347        /// try to not rely on this too much.
348        actually_rustdoc: bool [TRACKED],
349        /// Whether name resolver should resolve documentation links.
350        resolve_doc_links: ResolveDocLinks [TRACKED],
351
352        /// Control path trimming.
353        trimmed_def_paths: bool [TRACKED],
354
355        /// Specifications of codegen units / ThinLTO which are forced as a
356        /// result of parsing command line options. These are not necessarily
357        /// what rustc was invoked with, but massaged a bit to agree with
358        /// commands like `--emit llvm-ir` which they're often incompatible with
359        /// if we otherwise use the defaults of rustc.
360        #[rustc_lint_opt_deny_field_access("use `Session::codegen_units` instead of this field")]
361        cli_forced_codegen_units: Option<usize> [UNTRACKED],
362        #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
363        cli_forced_local_thinlto_off: bool [UNTRACKED],
364
365        /// Remap source path prefixes in all output (messages, object files, debug, etc.).
366        remap_path_prefix: Vec<(PathBuf, PathBuf)> [TRACKED_NO_CRATE_HASH],
367        /// Defines which scopes of paths should be remapped by `--remap-path-prefix`.
368        remap_path_scope: RemapPathScopeComponents [TRACKED_NO_CRATE_HASH],
369
370        /// Base directory containing the `library/` directory for the Rust standard library.
371        /// Right now it's always `$sysroot/lib/rustlib/src/rust`
372        /// (i.e. the `rustup` `rust-src` component).
373        ///
374        /// This directory is what the virtual `/rustc/$hash` is translated back to,
375        /// if Rust was built with path remapping to `/rustc/$hash` enabled
376        /// (the `rust.remap-debuginfo` option in `bootstrap.toml`).
377        real_rust_source_base_dir: Option<PathBuf> [TRACKED_NO_CRATE_HASH],
378
379        /// Base directory containing the `compiler/` directory for the rustc sources.
380        /// Right now it's always `$sysroot/lib/rustlib/rustc-src/rust`
381        /// (i.e. the `rustup` `rustc-dev` component).
382        ///
383        /// This directory is what the virtual `/rustc-dev/$hash` is translated back to,
384        /// if Rust was built with path remapping to `/rustc/$hash` enabled
385        /// (the `rust.remap-debuginfo` option in `bootstrap.toml`).
386        real_rustc_dev_source_base_dir: Option<PathBuf> [TRACKED_NO_CRATE_HASH],
387
388        edition: Edition [TRACKED],
389
390        /// `true` if we're emitting JSON blobs about each artifact produced
391        /// by the compiler.
392        json_artifact_notifications: bool [TRACKED],
393
394        /// `true` if we're emitting JSON timings with the start and end of
395        /// high-level compilation sections
396        json_timings: bool [UNTRACKED],
397
398        /// `true` if we're emitting a JSON blob containing the unused externs
399        json_unused_externs: JsonUnusedExterns [UNTRACKED],
400
401        /// `true` if we're emitting a JSON job containing a future-incompat report for lints
402        json_future_incompat: bool [TRACKED],
403
404        pretty: Option<PpMode> [UNTRACKED],
405
406        /// The (potentially remapped) working directory
407        #[rustc_lint_opt_deny_field_access("use `SourceMap::working_dir` instead of this field")]
408        working_dir: RealFileName [TRACKED],
409
410        color: ColorConfig [UNTRACKED],
411
412        verbose: bool [TRACKED_NO_CRATE_HASH],
413    }
414);
415
416#[derive(#[automatically_derived]
impl ::core::default::Default for CollectedOptions {
    #[inline]
    fn default() -> CollectedOptions {
        CollectedOptions {
            target_modifiers: ::core::default::Default::default(),
            mitigations: ::core::default::Default::default(),
        }
    }
}Default)]
417pub struct CollectedOptions {
418    pub target_modifiers: BTreeMap<OptionsTargetModifiers, String>,
419    pub mitigations: mitigation_coverage::MitigationCoverageMap,
420}
421
422macro_rules! setter_for {
423    // the allow/deny-mitigations options use collected/index instead of the cg, since they
424    // work across option groups
425    (allow_partial_mitigations, $struct_name:ident, $parse:ident) => {
426        pub(super) fn allow_partial_mitigations(
427            _cg: &mut super::$struct_name,
428            collected: &mut super::CollectedOptions,
429            v: Option<&str>,
430            index: usize,
431        ) -> bool {
432            collected.mitigations.handle_allowdeny_mitigation_option(v, index, true)
433        }
434    };
435    (deny_partial_mitigations, $struct_name:ident, $parse:ident) => {
436        pub(super) fn deny_partial_mitigations(
437            _cg: &mut super::$struct_name,
438            collected: &mut super::CollectedOptions,
439            v: Option<&str>,
440            index: usize,
441        ) -> bool {
442            collected.mitigations.handle_allowdeny_mitigation_option(v, index, false)
443        }
444    };
445    ($opt:ident, $struct_name:ident, $parse:ident) => {
446        pub(super) fn $opt(
447            cg: &mut super::$struct_name,
448            _collected: &mut super::CollectedOptions,
449            v: Option<&str>,
450            _index: usize,
451        ) -> bool {
452            super::parse::$parse(&mut redirect_field!(cg.$opt), v)
453        }
454    };
455}
456
457/// Defines all `CodegenOptions`/`DebuggingOptions` fields and parsers all at once. The goal of this
458/// macro is to define an interface that can be programmatically used by the option parser
459/// to initialize the struct without hardcoding field names all over the place.
460///
461/// The goal is to invoke this macro once with the correct fields, and then this macro generates all
462/// necessary code. The main gotcha of this macro is the `cgsetters` module which is a bunch of
463/// generated code to parse an option into its respective field in the struct. There are a few
464/// hand-written parsers for parsing specific types of values in this module.
465macro_rules! options {
466    (
467        $struct_name:ident,
468        $tmod_enum:ident,
469        $stat:ident,
470        $optmod:ident,
471        $prefix:expr,
472        $outputname:expr,
473
474        $(
475            $(#[$attr:meta])*
476            $opt:ident : $t:ty = (
477                $init:expr,
478                $parse:ident,
479                [$dep_tracking_marker:ident]
480                $( { TARGET_MODIFIER: $tmod_variant:ident } )?
481                $( { MITIGATION: $mitigation_variant:ident } )?
482                ,
483                $desc:literal
484                $(, removed: $removed:ident )?
485            ),
486        )*
487    ) => {
488        #[derive(Clone)]
489        #[rustc_lint_opt_ty]
490        pub struct $struct_name {
491            $(
492                $(#[$attr])*
493                pub $opt: $t,
494            )*
495        }
496
497        #[derive(PartialEq, Eq, PartialOrd, Ord, Debug, Copy, Clone, Encodable, BlobDecodable)]
498        pub enum $tmod_enum {
499            $(
500                $( $tmod_variant, )?
501            )*
502        }
503
504        impl $tmod_enum {
505            pub fn reparse(&self, _user_value: &str) -> ExtendedTargetModifierInfo {
506                match self {
507                    $(
508                        $(
509                            Self::$tmod_variant => {
510                                let mut parsed: $t = Default::default();
511                                let val = if _user_value.is_empty() { None } else { Some(_user_value) };
512                                parse::$parse(&mut parsed, val);
513                                ExtendedTargetModifierInfo {
514                                    prefix: $prefix.to_string(),
515                                    name: stringify!($opt).to_string().replace('_', "-"),
516                                    tech_value: format!("{:?}", parsed),
517                                }
518                            }
519                        )?
520                    )*
521
522                    #[allow(unreachable_patterns)]
523                    _ => panic!("unknown target modifier option: {:?}", *self)
524                }
525            }
526
527            pub fn is_target_modifier(flag_name: &str) -> bool {
528                match flag_name.replace('-', "_").as_str() {
529                    $(
530                        $(
531                            // Only expand for flags that have `TARGET_MODIFIER`.
532                            ${ignore($tmod_variant)}
533                            stringify!($opt) => true,
534                        )?
535                    )*
536                    _ => false,
537                }
538            }
539        }
540
541        impl Default for $struct_name {
542            fn default() -> $struct_name {
543                $struct_name {
544                    $(
545                        $opt: $init,
546                    )*
547                }
548            }
549        }
550
551        impl $struct_name {
552            pub fn build(
553                early_dcx: &EarlyDiagCtxt,
554                matches: &getopts::Matches,
555                target_modifiers: &mut CollectedOptions,
556            ) -> $struct_name {
557                build_options(early_dcx, matches, target_modifiers, $stat, $prefix, $outputname)
558            }
559
560            fn dep_tracking_hash(
561                &self,
562                for_crate_hash: bool,
563                error_format: ErrorOutputType,
564            ) -> Hash64 {
565                let mut sub_hashes = BTreeMap::new();
566                $(
567                    hash_opt!(
568                        $opt,
569                        &self.$opt,
570                        &mut sub_hashes,
571                        for_crate_hash,
572                        [$dep_tracking_marker]
573                    );
574                )*
575                let mut hasher = StableHasher::new();
576                dep_tracking::stable_hash(
577                    sub_hashes,
578                    &mut hasher,
579                    error_format,
580                    for_crate_hash,
581                );
582                hasher.finish()
583            }
584
585            pub fn gather_target_modifiers(
586                &self,
587                _mods: &mut Vec<TargetModifier>,
588                _tmod_vals: &BTreeMap<OptionsTargetModifiers, String>,
589            ) {
590                $(
591                    $(
592                        if self.$opt != $init {
593                            tmod_push_impl(
594                                OptionsTargetModifiers::$struct_name($tmod_enum::$tmod_variant),
595                                _tmod_vals,
596                                _mods,
597                            );
598                        }
599                    )?
600                )*
601            }
602        }
603
604        pub const $stat: OptionDescrs<$struct_name> = &[
605            $(
606                OptionDesc {
607                    name: stringify!($opt),
608                    setter: $optmod::$opt,
609                    type_desc: desc::$parse,
610                    desc: $desc,
611                    removed: None $( .or(Some(RemovedOption::$removed)) )?,
612                    tmod: None $( .or(Some(
613                        OptionsTargetModifiers::$struct_name($tmod_enum::$tmod_variant)
614                    )))?,
615                    mitigation: None $( .or(Some(
616                        mitigation_coverage::DeniedPartialMitigationKind::$mitigation_variant
617                    )))?,
618                },
619            )*
620        ];
621
622        mod $optmod {
623            $(
624                setter_for!($opt, $struct_name, $parse);
625            )*
626        }
627    }
628}
629
630impl CodegenOptions {
631    // JUSTIFICATION: defn of the suggested wrapper fn
632    #[allow(rustc::bad_opt_access)]
633    pub fn instrument_coverage(&self) -> InstrumentCoverage {
634        self.instrument_coverage
635    }
636}
637
638// Sometimes different options need to build a common structure.
639// That structure can be kept in one of the options' fields, the others become dummy.
640macro_rules! redirect_field {
641    ($cg:ident.link_arg) => {
642        $cg.link_args
643    };
644    ($cg:ident.pre_link_arg) => {
645        $cg.pre_link_args
646    };
647    ($cg:ident.$field:ident) => {
648        $cg.$field
649    };
650}
651
652type OptionSetter<O> = fn(&mut O, &mut CollectedOptions, v: Option<&str>, pos: usize) -> bool;
653type OptionDescrs<O> = &'static [OptionDesc<O>];
654
655/// Indicates whether a removed option should warn or error.
656enum RemovedOption {
657    #[allow(unused)] // we might want deprecated options that warn again in the future
658    Warn,
659    Err,
660}
661
662pub struct OptionDesc<O> {
663    name: &'static str,
664    setter: OptionSetter<O>,
665    // description for return value/type from mod desc
666    type_desc: &'static str,
667    // description for option from options table
668    desc: &'static str,
669    removed: Option<RemovedOption>,
670    tmod: Option<OptionsTargetModifiers>,
671    mitigation: Option<mitigation_coverage::DeniedPartialMitigationKind>,
672}
673
674impl<O> OptionDesc<O> {
675    pub fn name(&self) -> &'static str {
676        self.name
677    }
678
679    pub fn desc(&self) -> &'static str {
680        self.desc
681    }
682}
683
684fn build_options<O: Default>(
685    early_dcx: &EarlyDiagCtxt,
686    matches: &getopts::Matches,
687    collected_options: &mut CollectedOptions,
688    descrs: OptionDescrs<O>,
689    prefix: &str,
690    outputname: &str,
691) -> O {
692    let mut op = O::default();
693    for (index, option) in matches.opt_strs_pos(prefix) {
694        let (key, value) = match option.split_once('=') {
695            None => (option, None),
696            Some((k, v)) => (k.to_string(), Some(v)),
697        };
698
699        let option_to_lookup = key.replace('-', "_");
700        match descrs.iter().find(|opt_desc| opt_desc.name == option_to_lookup) {
701            Some(OptionDesc { name: _, setter, type_desc, desc, removed, tmod, mitigation }) => {
702                if let Some(removed) = removed {
703                    // deprecation works for prefixed options only
704                    if !!prefix.is_empty() {
    ::core::panicking::panic("assertion failed: !prefix.is_empty()")
};assert!(!prefix.is_empty());
705                    match removed {
706                        RemovedOption::Warn => {
707                            early_dcx.early_warn(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`-{0} {1}`: {2}", prefix, key,
                desc))
    })format!("`-{prefix} {key}`: {desc}"))
708                        }
709                        RemovedOption::Err => {
710                            early_dcx.early_fatal(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`-{0} {1}`: {2}", prefix, key,
                desc))
    })format!("`-{prefix} {key}`: {desc}"))
711                        }
712                    }
713                }
714                if !setter(&mut op, collected_options, value, index) {
715                    match value {
716                        None => early_dcx.early_fatal(
717                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} option `{1}` requires {2} (`-{3} {1}=<value>`)",
                outputname, key, type_desc, prefix))
    })format!(
718                                "{outputname} option `{key}` requires {type_desc} (`-{prefix} {key}=<value>`)"
719                            ),
720                        ),
721                        Some(value) => early_dcx.early_fatal(
722                            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("incorrect value `{0}` for {1} option `{2}` - {3} was expected",
                value, outputname, key, type_desc))
    })format!(
723                                "incorrect value `{value}` for {outputname} option `{key}` - {type_desc} was expected"
724                            ),
725                        ),
726                    }
727                }
728                if let Some(tmod) = *tmod {
729                    let v = value.map_or(String::new(), ToOwned::to_owned);
730                    collected_options.target_modifiers.insert(tmod, v);
731                }
732                if let Some(mitigation) = mitigation {
733                    collected_options.mitigations.reset_mitigation(*mitigation, index);
734                }
735            }
736            None => early_dcx.early_fatal(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("unknown {0} option: `{1}`",
                outputname, key))
    })format!("unknown {outputname} option: `{key}`")),
737        }
738    }
739    op
740}
741
742#[allow(non_upper_case_globals)]
743mod desc {
744    pub(crate) const parse_ignore: &str = "<ignored>"; // should not be user-visible
745    pub(crate) const parse_no_value: &str = "no value";
746    pub(crate) const parse_bool: &str =
747        "one of: `y`, `yes`, `on`, `true`, `n`, `no`, `off` or `false`";
748    pub(crate) const parse_opt_bool: &str = parse_bool;
749    pub(crate) const parse_string: &str = "a string";
750    pub(crate) const parse_opt_string: &str = parse_string;
751    pub(crate) const parse_string_push: &str = parse_string;
752    pub(crate) const parse_opt_pathbuf: &str = "a path";
753    pub(crate) const parse_list: &str = "a space-separated list of strings";
754    pub(crate) const parse_list_with_polarity: &str =
755        "a comma-separated list of strings, with elements beginning with + or -";
756    pub(crate) const parse_autodiff: &str = "a comma separated list of settings: `Enable`, `PrintSteps`, `PrintTA`, `PrintTAFn`, `PrintAA`, `PrintPerf`, `PrintModBefore`, `PrintModAfter`, `PrintModFinal`, `PrintPasses`, `NoPostopt`, `LooseTypes`, `Inline`, `NoTT`";
757    pub(crate) const parse_offload: &str =
758        "a comma separated list of settings: `Host=<Absolute-Path>`, `Device`, `Test`";
759    pub(crate) const parse_comma_list: &str = "a comma-separated list of strings";
760    pub(crate) const parse_opt_comma_list: &str = parse_comma_list;
761    pub(crate) const parse_number: &str = "a number";
762    pub(crate) const parse_opt_number: &str = parse_number;
763    pub(crate) const parse_frame_pointer: &str = "one of `true`/`yes`/`on`, `false`/`no`/`off`, or (with -Zunstable-options) `non-leaf` or `always`";
764    pub(crate) const parse_threads: &str = "a number or `sync`";
765    pub(crate) const parse_time_passes_format: &str = "`text` (default) or `json`";
766    pub(crate) const parse_passes: &str = "a space-separated list of passes, or `all`";
767    pub(crate) const parse_panic_strategy: &str = "either `unwind`, `abort`, or `immediate-abort`";
768    pub(crate) const parse_on_broken_pipe: &str = "either `kill`, `error`, or `inherit`";
769    pub(crate) const parse_patchable_function_entry: &str = "either two comma separated integers (total_nops,prefix_nops), with prefix_nops <= total_nops, or one integer (total_nops)";
770    pub(crate) const parse_opt_panic_strategy: &str = parse_panic_strategy;
771    pub(crate) const parse_relro_level: &str = "one of: `full`, `partial`, or `off`";
772    pub(crate) const parse_sanitizers: &str = "comma separated list of sanitizers: `address`, `cfi`, `dataflow`, `hwaddress`, `kcfi`, `kernel-address`, `kernel-hwaddress`, `leak`, `memory`, `memtag`, `safestack`, `shadow-call-stack`, `thread`, or 'realtime'";
773    pub(crate) const parse_sanitizer_memory_track_origins: &str = "0, 1, or 2";
774    pub(crate) const parse_cfguard: &str =
775        "either a boolean (`yes`, `no`, `on`, `off`, etc), `checks`, or `nochecks`";
776    pub(crate) const parse_cfprotection: &str = "`none`|`no`|`n` (default), `branch`, `return`, or `full`|`yes`|`y` (equivalent to `branch` and `return`)";
777    pub(crate) const parse_debuginfo: &str = "either an integer (0, 1, 2), `none`, `line-directives-only`, `line-tables-only`, `limited`, or `full`";
778    pub(crate) const parse_debuginfo_compression: &str = "one of `none`, `zlib`, or `zstd`";
779    pub(crate) const parse_mir_strip_debuginfo: &str =
780        "one of `none`, `locals-in-tiny-functions`, or `all-locals`";
781    pub(crate) const parse_collapse_macro_debuginfo: &str = "one of `no`, `external`, or `yes`";
782    pub(crate) const parse_strip: &str = "either `none`, `debuginfo`, or `symbols`";
783    pub(crate) const parse_linker_flavor: &str = ::rustc_target::spec::LinkerFlavorCli::one_of();
784    pub(crate) const parse_dump_mono_stats: &str = "`markdown` (default) or `json`";
785    pub(crate) const parse_instrument_coverage: &str = parse_bool;
786    pub(crate) const parse_coverage_options: &str = "`block` | `branch` | `condition`";
787    pub(crate) const parse_codegen_retag_options: &str =
788        "either no value or a comma-separated list of settings: `no-precise-im`, `no-precise-pin`";
789    pub(crate) 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`";
790    pub(crate) const parse_unpretty: &str = "`string` or `string=string`";
791    pub(crate) const parse_treat_err_as_bug: &str = "either no value or a non-negative number";
792    pub(crate) const parse_next_solver_config: &str =
793        "either `globally` (when used without an argument), `coherence` (default) or `no`";
794    pub(crate) const parse_lto: &str =
795        "either a boolean (`yes`, `no`, `on`, `off`, etc), `thin`, `fat`, or omitted";
796    pub(crate) const parse_linker_plugin_lto: &str =
797        "either a boolean (`yes`, `no`, `on`, `off`, etc), or the path to the linker plugin";
798    pub(crate) const parse_location_detail: &str = "either `none`, or a comma separated list of location details to track: `file`, `line`, or `column`";
799    pub(crate) const parse_fmt_debug: &str = "either `full`, `shallow`, or `none`";
800    pub(crate) const parse_switch_with_opt_path: &str =
801        "an optional path to the profiling data output directory";
802    pub(crate) const parse_merge_functions: &str =
803        "one of: `disabled`, `trampolines`, or `aliases`";
804    pub(crate) const parse_symbol_mangling_version: &str =
805        "one of: `legacy`, `v0` (RFC 2603), or `hashed`";
806    pub(crate) const parse_opt_symbol_visibility: &str =
807        "one of: `hidden`, `protected`, or `interposable`";
808    pub(crate) const parse_cargo_src_file_hash: &str =
809        "one of `blake3`, `md5`, `sha1`, or `sha256`";
810    pub(crate) const parse_src_file_hash: &str = "one of `md5`, `sha1`, or `sha256`";
811    pub(crate) const parse_relocation_model: &str =
812        "one of supported relocation models (`rustc --print relocation-models`)";
813    pub(crate) const parse_code_model: &str =
814        "one of supported code models (`rustc --print code-models`)";
815    pub(crate) const parse_tls_model: &str =
816        "one of supported TLS models (`rustc --print tls-models`)";
817    pub(crate) const parse_target_feature: &str = parse_string;
818    pub(crate) const parse_terminal_url: &str =
819        "either a boolean (`yes`, `no`, `on`, `off`, etc), or `auto`";
820    pub(crate) const parse_wasi_exec_model: &str = "either `command` or `reactor`";
821    pub(crate) const parse_split_debuginfo: &str =
822        "one of supported split-debuginfo modes (`off`, `packed`, or `unpacked`)";
823    pub(crate) const parse_split_dwarf_kind: &str =
824        "one of supported split dwarf modes (`split` or `single`)";
825    pub(crate) const parse_link_self_contained: &str = "one of: `y`, `yes`, `on`, `n`, `no`, `off`, or a list of enabled (`+` prefix) and disabled (`-` prefix) \
826        components: `crto`, `libc`, `unwind`, `linker`, `sanitizers`, `mingw`";
827    pub(crate) const parse_linker_features: &str =
828        "a list of enabled (`+` prefix) and disabled (`-` prefix) features: `lld`";
829    pub(crate) const parse_polonius: &str = "either no value or `legacy` (the default), or `next`";
830    pub(crate) const parse_annotate_moves: &str =
831        "either a boolean (`yes`, `no`, `on`, `off`, etc.), or a size limit in bytes";
832    pub(crate) const parse_stack_protector: &str =
833        "one of (`none` (default), `basic`, `strong`, or `all`)";
834    pub(crate) const parse_branch_protection: &str = "a `,` separated combination of `bti`, `gcs`, `pac-ret`, (optionally with `pc`, `b-key`, `leaf` if `pac-ret` is set)";
835    pub(crate) const parse_proc_macro_execution_strategy: &str =
836        "one of supported execution strategies (`same-thread`, or `cross-thread`)";
837    pub(crate) const parse_inlining_threshold: &str =
838        "either a boolean (`yes`, `no`, `on`, `off`, etc), or a non-negative number";
839    pub(crate) const parse_llvm_module_flag: &str = "<key>:<type>:<value>:<behavior>. Type must currently be `u32`. Behavior should be one of (`error`, `warning`, `require`, `override`, `append`, `appendunique`, `max`, `min`)";
840    pub(crate) const parse_function_return: &str = "`keep` or `thunk-extern`";
841    pub(crate) const parse_wasm_c_abi: &str = "`spec`";
842    pub(crate) const parse_mir_include_spans: &str =
843        "either a boolean (`yes`, `no`, `on`, `off`, etc), or `nll` (default: `nll`)";
844    pub(crate) const parse_align: &str = "a number that is a power of 2 between 1 and 2^29";
845    pub(crate) const parse_assert_incr_state: &str = "one of: `loaded`, `not-loaded`";
846    pub(crate) const parse_allow_partial_mitigations: &str =
847        super::mitigation_coverage::DeniedPartialMitigationKind::KINDS;
848    pub(crate) const parse_deny_partial_mitigations: &str =
849        super::mitigation_coverage::DeniedPartialMitigationKind::KINDS;
850    pub(crate) const parse_rust_version: &str = "a rust version (`xx.yy.zz`)";
851}
852
853pub mod parse {
854    use std::str::FromStr;
855
856    pub(crate) use super::*;
857    pub(crate) const MAX_THREADS_CAP: usize = 256;
858
859    /// Ignore the value. Used for removed options where we don't actually want to store
860    /// anything in the session.
861    pub(crate) fn parse_ignore(_slot: &mut (), _v: Option<&str>) -> bool {
862        true
863    }
864
865    /// This is for boolean options that don't take a value, and are true simply
866    /// by existing on the command-line.
867    ///
868    /// This style of option is deprecated, and is mainly used by old options
869    /// beginning with `no-`.
870    pub(crate) fn parse_no_value(slot: &mut bool, v: Option<&str>) -> bool {
871        match v {
872            None => {
873                *slot = true;
874                true
875            }
876            // Trying to specify a value is always forbidden.
877            Some(_) => false,
878        }
879    }
880
881    /// Use this for any boolean option that has a static default.
882    pub(crate) fn parse_bool(slot: &mut bool, v: Option<&str>) -> bool {
883        match v {
884            Some("y") | Some("yes") | Some("on") | Some("true") | None => {
885                *slot = true;
886                true
887            }
888            Some("n") | Some("no") | Some("off") | Some("false") => {
889                *slot = false;
890                true
891            }
892            _ => false,
893        }
894    }
895
896    /// Use this for any boolean option that lacks a static default. (The
897    /// actions taken when such an option is not specified will depend on
898    /// other factors, such as other options, or target options.)
899    pub(crate) fn parse_opt_bool(slot: &mut Option<bool>, v: Option<&str>) -> bool {
900        match v {
901            Some("y") | Some("yes") | Some("on") | Some("true") | None => {
902                *slot = Some(true);
903                true
904            }
905            Some("n") | Some("no") | Some("off") | Some("false") => {
906                *slot = Some(false);
907                true
908            }
909            _ => false,
910        }
911    }
912
913    /// Parses whether polonius is enabled, and if so, which version.
914    pub(crate) fn parse_polonius(slot: &mut Polonius, v: Option<&str>) -> bool {
915        match v {
916            Some("legacy") | None => {
917                *slot = Polonius::Legacy;
918                true
919            }
920            Some("next") => {
921                *slot = Polonius::Next;
922                true
923            }
924            _ => false,
925        }
926    }
927
928    pub(crate) fn parse_annotate_moves(slot: &mut AnnotateMoves, v: Option<&str>) -> bool {
929        let mut bslot = false;
930        let mut nslot = 0u64;
931
932        *slot = match v {
933            // No value provided: -Z annotate-moves (enable with default limit)
934            None => AnnotateMoves::Enabled(None),
935            // Explicit boolean value provided: -Z annotate-moves=yes/no
936            s @ Some(_) if parse_bool(&mut bslot, s) => {
937                if bslot {
938                    AnnotateMoves::Enabled(None)
939                } else {
940                    AnnotateMoves::Disabled
941                }
942            }
943            // With numeric limit provided: -Z annotate-moves=1234
944            s @ Some(_) if parse_number(&mut nslot, s) => AnnotateMoves::Enabled(Some(nslot)),
945            _ => return false,
946        };
947
948        true
949    }
950
951    /// Use this for any string option that has a static default.
952    pub(crate) fn parse_string(slot: &mut String, v: Option<&str>) -> bool {
953        match v {
954            Some(s) => {
955                *slot = s.to_string();
956                true
957            }
958            None => false,
959        }
960    }
961
962    /// Use this for any string option that lacks a static default.
963    pub(crate) fn parse_opt_string(slot: &mut Option<String>, v: Option<&str>) -> bool {
964        match v {
965            Some(s) => {
966                *slot = Some(s.to_string());
967                true
968            }
969            None => false,
970        }
971    }
972
973    pub(crate) fn parse_opt_pathbuf(slot: &mut Option<PathBuf>, v: Option<&str>) -> bool {
974        match v {
975            Some(s) => {
976                *slot = Some(PathBuf::from(s));
977                true
978            }
979            None => false,
980        }
981    }
982
983    pub(crate) fn parse_string_push(slot: &mut Vec<String>, v: Option<&str>) -> bool {
984        match v {
985            Some(s) => {
986                slot.push(s.to_string());
987                true
988            }
989            None => false,
990        }
991    }
992
993    pub(crate) fn parse_list(slot: &mut Vec<String>, v: Option<&str>) -> bool {
994        match v {
995            Some(s) => {
996                slot.extend(s.split_whitespace().map(|s| s.to_string()));
997                true
998            }
999            None => false,
1000        }
1001    }
1002
1003    pub(crate) fn parse_list_with_polarity(
1004        slot: &mut Vec<(String, bool)>,
1005        v: Option<&str>,
1006    ) -> bool {
1007        match v {
1008            Some(s) => {
1009                for s in s.split(',') {
1010                    let Some(pass_name) = s.strip_prefix(&['+', '-'][..]) else { return false };
1011                    slot.push((pass_name.to_string(), &s[..1] == "+"));
1012                }
1013                true
1014            }
1015            None => false,
1016        }
1017    }
1018
1019    pub(crate) fn parse_fmt_debug(opt: &mut FmtDebug, v: Option<&str>) -> bool {
1020        *opt = match v {
1021            Some("full") => FmtDebug::Full,
1022            Some("shallow") => FmtDebug::Shallow,
1023            Some("none") => FmtDebug::None,
1024            _ => return false,
1025        };
1026        true
1027    }
1028
1029    pub(crate) fn parse_location_detail(ld: &mut LocationDetail, v: Option<&str>) -> bool {
1030        if let Some(v) = v {
1031            ld.line = false;
1032            ld.file = false;
1033            ld.column = false;
1034            if v == "none" {
1035                return true;
1036            }
1037            for s in v.split(',') {
1038                match s {
1039                    "file" => ld.file = true,
1040                    "line" => ld.line = true,
1041                    "column" => ld.column = true,
1042                    _ => return false,
1043                }
1044            }
1045            true
1046        } else {
1047            false
1048        }
1049    }
1050
1051    pub(crate) fn parse_comma_list(slot: &mut Vec<String>, v: Option<&str>) -> bool {
1052        match v {
1053            Some(s) => {
1054                let mut v: Vec<_> = s.split(',').map(|s| s.to_string()).collect();
1055                v.sort_unstable();
1056                *slot = v;
1057                true
1058            }
1059            None => false,
1060        }
1061    }
1062
1063    pub(crate) fn parse_opt_comma_list(slot: &mut Option<Vec<String>>, v: Option<&str>) -> bool {
1064        match v {
1065            Some(s) => {
1066                let mut v: Vec<_> = s.split(',').map(|s| s.to_string()).collect();
1067                v.sort_unstable();
1068                *slot = Some(v);
1069                true
1070            }
1071            None => false,
1072        }
1073    }
1074
1075    pub(crate) fn parse_threads(slot: &mut Option<usize>, v: Option<&str>) -> bool {
1076        let Some(s) = v else { return false };
1077        if s == "sync" {
1078            // Enable synchronization despite only using one thread.
1079            *slot = Some(1);
1080            return true;
1081        }
1082        let n = match s.parse().ok() {
1083            Some(0) => std::thread::available_parallelism().map_or(1, NonZero::<usize>::get),
1084            Some(i) => i,
1085            None => return false,
1086        };
1087        // We want to cap the number of threads here to avoid large numbers like 999999 and compiler panics.
1088        // This solution was suggested here https://github.com/rust-lang/rust/issues/117638#issuecomment-1800925067
1089        let n = n.min(MAX_THREADS_CAP);
1090        *slot = (n > 1).then_some(n); // Enable synchronization if we're using more than one thread.
1091        true
1092    }
1093
1094    /// Use this for any numeric option that has a static default.
1095    pub(crate) fn parse_number<T: Copy + FromStr>(slot: &mut T, v: Option<&str>) -> bool {
1096        match v.and_then(|s| s.parse().ok()) {
1097            Some(i) => {
1098                *slot = i;
1099                true
1100            }
1101            None => false,
1102        }
1103    }
1104
1105    /// Use this for any numeric option that lacks a static default.
1106    pub(crate) fn parse_opt_number<T: Copy + FromStr>(
1107        slot: &mut Option<T>,
1108        v: Option<&str>,
1109    ) -> bool {
1110        match v {
1111            Some(s) => {
1112                *slot = s.parse().ok();
1113                slot.is_some()
1114            }
1115            None => false,
1116        }
1117    }
1118
1119    pub(crate) fn parse_frame_pointer(slot: &mut FramePointer, v: Option<&str>) -> bool {
1120        let mut yes = false;
1121        match v {
1122            _ if parse_bool(&mut yes, v) && yes => slot.ratchet(FramePointer::Always),
1123            _ if parse_bool(&mut yes, v) => slot.ratchet(FramePointer::MayOmit),
1124            Some("always") => slot.ratchet(FramePointer::Always),
1125            Some("non-leaf") => slot.ratchet(FramePointer::NonLeaf),
1126            _ => return false,
1127        };
1128        true
1129    }
1130
1131    pub(crate) fn parse_passes(slot: &mut Passes, v: Option<&str>) -> bool {
1132        match v {
1133            Some("all") => {
1134                *slot = Passes::All;
1135                true
1136            }
1137            v => {
1138                let mut passes = ::alloc::vec::Vec::new()vec![];
1139                if parse_list(&mut passes, v) {
1140                    slot.extend(passes);
1141                    true
1142                } else {
1143                    false
1144                }
1145            }
1146        }
1147    }
1148
1149    pub(crate) fn parse_opt_panic_strategy(
1150        slot: &mut Option<PanicStrategy>,
1151        v: Option<&str>,
1152    ) -> bool {
1153        match v {
1154            Some("unwind") => *slot = Some(PanicStrategy::Unwind),
1155            Some("abort") => *slot = Some(PanicStrategy::Abort),
1156            Some("immediate-abort") => *slot = Some(PanicStrategy::ImmediateAbort),
1157            _ => return false,
1158        }
1159        true
1160    }
1161
1162    pub(crate) fn parse_panic_strategy(slot: &mut PanicStrategy, v: Option<&str>) -> bool {
1163        match v {
1164            Some("unwind") => *slot = PanicStrategy::Unwind,
1165            Some("abort") => *slot = PanicStrategy::Abort,
1166            Some("immediate-abort") => *slot = PanicStrategy::ImmediateAbort,
1167            _ => return false,
1168        }
1169        true
1170    }
1171
1172    pub(crate) fn parse_on_broken_pipe(slot: &mut OnBrokenPipe, v: Option<&str>) -> bool {
1173        match v {
1174            // OnBrokenPipe::Default can't be explicitly specified
1175            Some("kill") => *slot = OnBrokenPipe::Kill,
1176            Some("error") => *slot = OnBrokenPipe::Error,
1177            Some("inherit") => *slot = OnBrokenPipe::Inherit,
1178            _ => return false,
1179        }
1180        true
1181    }
1182
1183    pub(crate) fn parse_patchable_function_entry(
1184        slot: &mut PatchableFunctionEntry,
1185        v: Option<&str>,
1186    ) -> bool {
1187        let mut total_nops = 0;
1188        let mut prefix_nops = 0;
1189
1190        if !parse_number(&mut total_nops, v) {
1191            let parts = v.and_then(|v| v.split_once(',')).unzip();
1192            if !parse_number(&mut total_nops, parts.0) {
1193                return false;
1194            }
1195            if !parse_number(&mut prefix_nops, parts.1) {
1196                return false;
1197            }
1198        }
1199
1200        if let Some(pfe) =
1201            PatchableFunctionEntry::from_total_and_prefix_nops(total_nops, prefix_nops)
1202        {
1203            *slot = pfe;
1204            return true;
1205        }
1206        false
1207    }
1208
1209    pub(crate) fn parse_relro_level(slot: &mut Option<RelroLevel>, v: Option<&str>) -> bool {
1210        match v {
1211            Some(s) => match s.parse::<RelroLevel>() {
1212                Ok(level) => *slot = Some(level),
1213                _ => return false,
1214            },
1215            _ => return false,
1216        }
1217        true
1218    }
1219
1220    pub(crate) fn parse_sanitizers(slot: &mut SanitizerSet, v: Option<&str>) -> bool {
1221        if let Some(v) = v {
1222            for s in v.split(',') {
1223                *slot |= match s {
1224                    "address" => SanitizerSet::ADDRESS,
1225                    "cfi" => SanitizerSet::CFI,
1226                    "dataflow" => SanitizerSet::DATAFLOW,
1227                    "kcfi" => SanitizerSet::KCFI,
1228                    "kernel-address" => SanitizerSet::KERNELADDRESS,
1229                    "kernel-hwaddress" => SanitizerSet::KERNELHWADDRESS,
1230                    "leak" => SanitizerSet::LEAK,
1231                    "memory" => SanitizerSet::MEMORY,
1232                    "memtag" => SanitizerSet::MEMTAG,
1233                    "shadow-call-stack" => SanitizerSet::SHADOWCALLSTACK,
1234                    "thread" => SanitizerSet::THREAD,
1235                    "hwaddress" => SanitizerSet::HWADDRESS,
1236                    "safestack" => SanitizerSet::SAFESTACK,
1237                    "realtime" => SanitizerSet::REALTIME,
1238                    _ => return false,
1239                }
1240            }
1241            true
1242        } else {
1243            false
1244        }
1245    }
1246
1247    pub(crate) fn parse_sanitizer_memory_track_origins(slot: &mut usize, v: Option<&str>) -> bool {
1248        match v {
1249            Some("2") | None => {
1250                *slot = 2;
1251                true
1252            }
1253            Some("1") => {
1254                *slot = 1;
1255                true
1256            }
1257            Some("0") => {
1258                *slot = 0;
1259                true
1260            }
1261            Some(_) => false,
1262        }
1263    }
1264
1265    pub(crate) fn parse_strip(slot: &mut Strip, v: Option<&str>) -> bool {
1266        match v {
1267            Some("none") => *slot = Strip::None,
1268            Some("debuginfo") => *slot = Strip::Debuginfo,
1269            Some("symbols") => *slot = Strip::Symbols,
1270            _ => return false,
1271        }
1272        true
1273    }
1274
1275    pub(crate) fn parse_cfguard(slot: &mut CFGuard, v: Option<&str>) -> bool {
1276        if v.is_some() {
1277            let mut bool_arg = None;
1278            if parse_opt_bool(&mut bool_arg, v) {
1279                *slot = if bool_arg.unwrap() { CFGuard::Checks } else { CFGuard::Disabled };
1280                return true;
1281            }
1282        }
1283
1284        *slot = match v {
1285            None => CFGuard::Checks,
1286            Some("checks") => CFGuard::Checks,
1287            Some("nochecks") => CFGuard::NoChecks,
1288            Some(_) => return false,
1289        };
1290        true
1291    }
1292
1293    pub(crate) fn parse_cfprotection(slot: &mut CFProtection, v: Option<&str>) -> bool {
1294        if v.is_some() {
1295            let mut bool_arg = None;
1296            if parse_opt_bool(&mut bool_arg, v) {
1297                *slot = if bool_arg.unwrap() { CFProtection::Full } else { CFProtection::None };
1298                return true;
1299            }
1300        }
1301
1302        *slot = match v {
1303            None | Some("none") => CFProtection::None,
1304            Some("branch") => CFProtection::Branch,
1305            Some("return") => CFProtection::Return,
1306            Some("full") => CFProtection::Full,
1307            Some(_) => return false,
1308        };
1309        true
1310    }
1311
1312    pub(crate) fn parse_debuginfo(slot: &mut DebugInfo, v: Option<&str>) -> bool {
1313        match v {
1314            Some("0") | Some("none") => *slot = DebugInfo::None,
1315            Some("line-directives-only") => *slot = DebugInfo::LineDirectivesOnly,
1316            Some("line-tables-only") => *slot = DebugInfo::LineTablesOnly,
1317            Some("1") | Some("limited") => *slot = DebugInfo::Limited,
1318            Some("2") | Some("full") => *slot = DebugInfo::Full,
1319            _ => return false,
1320        }
1321        true
1322    }
1323
1324    pub(crate) fn parse_debuginfo_compression(
1325        slot: &mut DebugInfoCompression,
1326        v: Option<&str>,
1327    ) -> bool {
1328        match v {
1329            Some("none") => *slot = DebugInfoCompression::None,
1330            Some("zlib") => *slot = DebugInfoCompression::Zlib,
1331            Some("zstd") => *slot = DebugInfoCompression::Zstd,
1332            _ => return false,
1333        };
1334        true
1335    }
1336
1337    pub(crate) fn parse_mir_strip_debuginfo(slot: &mut MirStripDebugInfo, v: Option<&str>) -> bool {
1338        match v {
1339            Some("none") => *slot = MirStripDebugInfo::None,
1340            Some("locals-in-tiny-functions") => *slot = MirStripDebugInfo::LocalsInTinyFunctions,
1341            Some("all-locals") => *slot = MirStripDebugInfo::AllLocals,
1342            _ => return false,
1343        };
1344        true
1345    }
1346
1347    pub(crate) fn parse_linker_flavor(slot: &mut Option<LinkerFlavorCli>, v: Option<&str>) -> bool {
1348        match v.and_then(|v| LinkerFlavorCli::from_str(v).ok()) {
1349            Some(lf) => *slot = Some(lf),
1350            _ => return false,
1351        }
1352        true
1353    }
1354
1355    pub(crate) fn parse_opt_symbol_visibility(
1356        slot: &mut Option<SymbolVisibility>,
1357        v: Option<&str>,
1358    ) -> bool {
1359        if let Some(v) = v {
1360            if let Ok(vis) = SymbolVisibility::from_str(v) {
1361                *slot = Some(vis);
1362            } else {
1363                return false;
1364            }
1365        }
1366        true
1367    }
1368
1369    pub(crate) fn parse_unpretty(slot: &mut Option<String>, v: Option<&str>) -> bool {
1370        match v {
1371            None => false,
1372            Some(s) if s.split('=').count() <= 2 => {
1373                *slot = Some(s.to_string());
1374                true
1375            }
1376            _ => false,
1377        }
1378    }
1379
1380    pub(crate) fn parse_time_passes_format(slot: &mut TimePassesFormat, v: Option<&str>) -> bool {
1381        match v {
1382            None => true,
1383            Some("json") => {
1384                *slot = TimePassesFormat::Json;
1385                true
1386            }
1387            Some("text") => {
1388                *slot = TimePassesFormat::Text;
1389                true
1390            }
1391            Some(_) => false,
1392        }
1393    }
1394
1395    pub(crate) fn parse_dump_mono_stats(slot: &mut DumpMonoStatsFormat, v: Option<&str>) -> bool {
1396        match v {
1397            None => true,
1398            Some("json") => {
1399                *slot = DumpMonoStatsFormat::Json;
1400                true
1401            }
1402            Some("markdown") => {
1403                *slot = DumpMonoStatsFormat::Markdown;
1404                true
1405            }
1406            Some(_) => false,
1407        }
1408    }
1409
1410    pub(crate) fn parse_offload(slot: &mut Vec<Offload>, v: Option<&str>) -> bool {
1411        let Some(v) = v else {
1412            *slot = ::alloc::vec::Vec::new()vec![];
1413            return true;
1414        };
1415        let mut v: Vec<&str> = v.split(",").collect();
1416        v.sort_unstable();
1417        for &val in v.iter() {
1418            // Split each entry on '=' if it has an argument
1419            let (key, arg) = match val.split_once('=') {
1420                Some((k, a)) => (k, Some(a)),
1421                None => (val, None),
1422            };
1423
1424            let variant = match key {
1425                "Host" => {
1426                    if let Some(p) = arg {
1427                        Offload::Host(p.to_string())
1428                    } else {
1429                        return false;
1430                    }
1431                }
1432                "Device" => {
1433                    if let Some(_) = arg {
1434                        // Device does not accept a value
1435                        return false;
1436                    }
1437                    Offload::Device
1438                }
1439                "Test" => {
1440                    if let Some(_) = arg {
1441                        // Test does not accept a value
1442                        return false;
1443                    }
1444                    Offload::Test
1445                }
1446                _ => {
1447                    // FIXME(ZuseZ4): print an error saying which value is not recognized
1448                    return false;
1449                }
1450            };
1451            slot.push(variant);
1452        }
1453
1454        true
1455    }
1456
1457    pub(crate) fn parse_autodiff(slot: &mut Vec<AutoDiff>, v: Option<&str>) -> bool {
1458        let Some(v) = v else {
1459            *slot = ::alloc::vec::Vec::new()vec![];
1460            return true;
1461        };
1462        let mut v: Vec<&str> = v.split(",").collect();
1463        v.sort_unstable();
1464        for &val in v.iter() {
1465            // Split each entry on '=' if it has an argument
1466            let (key, arg) = match val.split_once('=') {
1467                Some((k, a)) => (k, Some(a)),
1468                None => (val, None),
1469            };
1470
1471            let variant = match key {
1472                "Enable" => AutoDiff::Enable,
1473                "PrintTA" => AutoDiff::PrintTA,
1474                "PrintTAFn" => {
1475                    if let Some(fun) = arg {
1476                        AutoDiff::PrintTAFn(fun.to_string())
1477                    } else {
1478                        return false;
1479                    }
1480                }
1481                "PrintAA" => AutoDiff::PrintAA,
1482                "PrintPerf" => AutoDiff::PrintPerf,
1483                "PrintSteps" => AutoDiff::PrintSteps,
1484                "PrintModBefore" => AutoDiff::PrintModBefore,
1485                "PrintModAfter" => AutoDiff::PrintModAfter,
1486                "PrintModFinal" => AutoDiff::PrintModFinal,
1487                "NoPostopt" => AutoDiff::NoPostopt,
1488                "PrintPasses" => AutoDiff::PrintPasses,
1489                "LooseTypes" => AutoDiff::LooseTypes,
1490                "Inline" => AutoDiff::Inline,
1491                "NoTT" => AutoDiff::NoTT,
1492                _ => {
1493                    // FIXME(ZuseZ4): print an error saying which value is not recognized
1494                    return false;
1495                }
1496            };
1497            slot.push(variant);
1498        }
1499
1500        true
1501    }
1502
1503    pub(crate) fn parse_instrument_coverage(
1504        slot: &mut InstrumentCoverage,
1505        v: Option<&str>,
1506    ) -> bool {
1507        if v.is_some() {
1508            let mut bool_arg = false;
1509            if parse_bool(&mut bool_arg, v) {
1510                *slot = if bool_arg { InstrumentCoverage::Yes } else { InstrumentCoverage::No };
1511                return true;
1512            }
1513        }
1514
1515        let Some(v) = v else {
1516            *slot = InstrumentCoverage::Yes;
1517            return true;
1518        };
1519
1520        // Parse values that have historically been accepted by stable compilers,
1521        // even though they're currently just aliases for boolean values.
1522        *slot = match v {
1523            "all" => InstrumentCoverage::Yes,
1524            "0" => InstrumentCoverage::No,
1525            _ => return false,
1526        };
1527        true
1528    }
1529
1530    pub(crate) fn parse_codegen_retag_options(
1531        slot: &mut Option<CodegenRetagOptions>,
1532        v: Option<&str>,
1533    ) -> bool {
1534        let mut no_precise_im = false;
1535        let mut no_precise_pin = false;
1536        if let Some(opt_list) = v.map(|s| s.split(',')) {
1537            for opt in opt_list {
1538                match opt {
1539                    "no-precise-im" => {
1540                        no_precise_im = true;
1541                    }
1542                    "no-precise-pin" => {
1543                        no_precise_pin = true;
1544                    }
1545                    _ => return false,
1546                }
1547            }
1548        }
1549        *slot = Some(CodegenRetagOptions { no_precise_im, no_precise_pin });
1550        true
1551    }
1552
1553    pub(crate) fn parse_coverage_options(slot: &mut CoverageOptions, v: Option<&str>) -> bool {
1554        let Some(v) = v else { return true };
1555
1556        for option in v.split(',') {
1557            match option {
1558                "block" => slot.level = CoverageLevel::Block,
1559                "branch" => slot.level = CoverageLevel::Branch,
1560                "condition" => slot.level = CoverageLevel::Condition,
1561                "discard-all-spans-in-codegen" => slot.discard_all_spans_in_codegen = true,
1562                _ => return false,
1563            }
1564        }
1565        true
1566    }
1567
1568    pub(crate) fn parse_instrument_xray(
1569        slot: &mut Option<InstrumentXRay>,
1570        v: Option<&str>,
1571    ) -> bool {
1572        if v.is_some() {
1573            let mut bool_arg = None;
1574            if parse_opt_bool(&mut bool_arg, v) {
1575                *slot = if bool_arg.unwrap() { Some(InstrumentXRay::default()) } else { None };
1576                return true;
1577            }
1578        }
1579
1580        let options = slot.get_or_insert_default();
1581        let mut seen_always = false;
1582        let mut seen_never = false;
1583        let mut seen_ignore_loops = false;
1584        let mut seen_instruction_threshold = false;
1585        let mut seen_skip_entry = false;
1586        let mut seen_skip_exit = false;
1587        for option in v.into_iter().flat_map(|v| v.split(',')) {
1588            match option {
1589                "always" if !seen_always && !seen_never => {
1590                    options.always = true;
1591                    options.never = false;
1592                    seen_always = true;
1593                }
1594                "never" if !seen_never && !seen_always => {
1595                    options.never = true;
1596                    options.always = false;
1597                    seen_never = true;
1598                }
1599                "ignore-loops" if !seen_ignore_loops => {
1600                    options.ignore_loops = true;
1601                    seen_ignore_loops = true;
1602                }
1603                option
1604                    if option.starts_with("instruction-threshold")
1605                        && !seen_instruction_threshold =>
1606                {
1607                    let Some(("instruction-threshold", n)) = option.split_once('=') else {
1608                        return false;
1609                    };
1610                    match n.parse() {
1611                        Ok(n) => options.instruction_threshold = Some(n),
1612                        Err(_) => return false,
1613                    }
1614                    seen_instruction_threshold = true;
1615                }
1616                "skip-entry" if !seen_skip_entry => {
1617                    options.skip_entry = true;
1618                    seen_skip_entry = true;
1619                }
1620                "skip-exit" if !seen_skip_exit => {
1621                    options.skip_exit = true;
1622                    seen_skip_exit = true;
1623                }
1624                _ => return false,
1625            }
1626        }
1627        true
1628    }
1629
1630    pub(crate) fn parse_treat_err_as_bug(
1631        slot: &mut Option<NonZero<usize>>,
1632        v: Option<&str>,
1633    ) -> bool {
1634        match v {
1635            Some(s) => match s.parse() {
1636                Ok(val) => {
1637                    *slot = Some(val);
1638                    true
1639                }
1640                Err(e) => {
1641                    *slot = None;
1642                    e.kind() == &IntErrorKind::Zero
1643                }
1644            },
1645            None => {
1646                *slot = NonZero::new(1);
1647                true
1648            }
1649        }
1650    }
1651
1652    pub(crate) fn parse_next_solver_config(slot: &mut NextSolverConfig, v: Option<&str>) -> bool {
1653        if let Some(config) = v {
1654            *slot = match config {
1655                "no" => NextSolverConfig { coherence: false, globally: false },
1656                "coherence" => NextSolverConfig { coherence: true, globally: false },
1657                "globally" => NextSolverConfig { coherence: true, globally: true },
1658                _ => return false,
1659            };
1660        } else {
1661            *slot = NextSolverConfig { coherence: true, globally: true };
1662        }
1663
1664        true
1665    }
1666
1667    pub(crate) fn parse_lto(slot: &mut LtoCli, v: Option<&str>) -> bool {
1668        if v.is_some() {
1669            let mut bool_arg = None;
1670            if parse_opt_bool(&mut bool_arg, v) {
1671                *slot = if bool_arg.unwrap() { LtoCli::Yes } else { LtoCli::No };
1672                return true;
1673            }
1674        }
1675
1676        *slot = match v {
1677            None => LtoCli::NoParam,
1678            Some("thin") => LtoCli::Thin,
1679            Some("fat") => LtoCli::Fat,
1680            Some(_) => return false,
1681        };
1682        true
1683    }
1684
1685    pub(crate) fn parse_linker_plugin_lto(slot: &mut LinkerPluginLto, v: Option<&str>) -> bool {
1686        if v.is_some() {
1687            let mut bool_arg = None;
1688            if parse_opt_bool(&mut bool_arg, v) {
1689                *slot = if bool_arg.unwrap() {
1690                    LinkerPluginLto::LinkerPluginAuto
1691                } else {
1692                    LinkerPluginLto::Disabled
1693                };
1694                return true;
1695            }
1696        }
1697
1698        *slot = match v {
1699            None => LinkerPluginLto::LinkerPluginAuto,
1700            Some(path) => LinkerPluginLto::LinkerPlugin(PathBuf::from(path)),
1701        };
1702        true
1703    }
1704
1705    pub(crate) fn parse_switch_with_opt_path(
1706        slot: &mut SwitchWithOptPath,
1707        v: Option<&str>,
1708    ) -> bool {
1709        *slot = match v {
1710            None => SwitchWithOptPath::Enabled(None),
1711            Some(path) => SwitchWithOptPath::Enabled(Some(PathBuf::from(path))),
1712        };
1713        true
1714    }
1715
1716    pub(crate) fn parse_merge_functions(
1717        slot: &mut Option<MergeFunctions>,
1718        v: Option<&str>,
1719    ) -> bool {
1720        match v.and_then(|s| MergeFunctions::from_str(s).ok()) {
1721            Some(mergefunc) => *slot = Some(mergefunc),
1722            _ => return false,
1723        }
1724        true
1725    }
1726
1727    pub(crate) fn parse_relocation_model(slot: &mut Option<RelocModel>, v: Option<&str>) -> bool {
1728        match v.and_then(|s| RelocModel::from_str(s).ok()) {
1729            Some(relocation_model) => *slot = Some(relocation_model),
1730            None if v == Some("default") => *slot = None,
1731            _ => return false,
1732        }
1733        true
1734    }
1735
1736    pub(crate) fn parse_code_model(slot: &mut Option<CodeModel>, v: Option<&str>) -> bool {
1737        match v.and_then(|s| CodeModel::from_str(s).ok()) {
1738            Some(code_model) => *slot = Some(code_model),
1739            _ => return false,
1740        }
1741        true
1742    }
1743
1744    pub(crate) fn parse_tls_model(slot: &mut Option<TlsModel>, v: Option<&str>) -> bool {
1745        match v.and_then(|s| TlsModel::from_str(s).ok()) {
1746            Some(tls_model) => *slot = Some(tls_model),
1747            _ => return false,
1748        }
1749        true
1750    }
1751
1752    pub(crate) fn parse_terminal_url(slot: &mut TerminalUrl, v: Option<&str>) -> bool {
1753        *slot = match v {
1754            Some("on" | "" | "yes" | "y") | None => TerminalUrl::Yes,
1755            Some("off" | "no" | "n") => TerminalUrl::No,
1756            Some("auto") => TerminalUrl::Auto,
1757            _ => return false,
1758        };
1759        true
1760    }
1761
1762    pub(crate) fn parse_symbol_mangling_version(
1763        slot: &mut Option<SymbolManglingVersion>,
1764        v: Option<&str>,
1765    ) -> bool {
1766        *slot = match v {
1767            Some("legacy") => Some(SymbolManglingVersion::Legacy),
1768            Some("v0") => Some(SymbolManglingVersion::V0),
1769            Some("hashed") => Some(SymbolManglingVersion::Hashed),
1770            _ => return false,
1771        };
1772        true
1773    }
1774
1775    pub(crate) fn parse_src_file_hash(
1776        slot: &mut Option<SourceFileHashAlgorithm>,
1777        v: Option<&str>,
1778    ) -> bool {
1779        match v.and_then(|s| SourceFileHashAlgorithm::from_str(s).ok()) {
1780            Some(hash_kind) => *slot = Some(hash_kind),
1781            _ => return false,
1782        }
1783        true
1784    }
1785
1786    pub(crate) fn parse_cargo_src_file_hash(
1787        slot: &mut Option<SourceFileHashAlgorithm>,
1788        v: Option<&str>,
1789    ) -> bool {
1790        match v.and_then(|s| SourceFileHashAlgorithm::from_str(s).ok()) {
1791            Some(hash_kind) => {
1792                *slot = Some(hash_kind);
1793            }
1794            _ => return false,
1795        }
1796        true
1797    }
1798
1799    pub(crate) fn parse_target_feature(slot: &mut String, v: Option<&str>) -> bool {
1800        match v {
1801            Some(s) => {
1802                if !slot.is_empty() {
1803                    slot.push(',');
1804                }
1805                slot.push_str(s);
1806                true
1807            }
1808            None => false,
1809        }
1810    }
1811
1812    pub(crate) fn parse_link_self_contained(slot: &mut LinkSelfContained, v: Option<&str>) -> bool {
1813        // Whenever `-C link-self-contained` is passed without a value, it's an opt-in
1814        // just like `parse_opt_bool`, the historical value of this flag.
1815        //
1816        // 1. Parse historical single bool values
1817        let s = v.unwrap_or("y");
1818        match s {
1819            "y" | "yes" | "on" => {
1820                slot.set_all_explicitly(true);
1821                return true;
1822            }
1823            "n" | "no" | "off" => {
1824                slot.set_all_explicitly(false);
1825                return true;
1826            }
1827            _ => {}
1828        }
1829
1830        // 2. Parse a list of enabled and disabled components.
1831        for comp in s.split(',') {
1832            if slot.handle_cli_component(comp).is_none() {
1833                return false;
1834            }
1835        }
1836
1837        true
1838    }
1839
1840    /// Parse a comma-separated list of enabled and disabled linker features.
1841    pub(crate) fn parse_linker_features(slot: &mut LinkerFeaturesCli, v: Option<&str>) -> bool {
1842        match v {
1843            Some(s) => {
1844                for feature in s.split(',') {
1845                    if slot.handle_cli_feature(feature).is_none() {
1846                        return false;
1847                    }
1848                }
1849
1850                true
1851            }
1852            None => false,
1853        }
1854    }
1855
1856    pub(crate) fn parse_wasi_exec_model(slot: &mut Option<WasiExecModel>, v: Option<&str>) -> bool {
1857        match v {
1858            Some("command") => *slot = Some(WasiExecModel::Command),
1859            Some("reactor") => *slot = Some(WasiExecModel::Reactor),
1860            _ => return false,
1861        }
1862        true
1863    }
1864
1865    pub(crate) fn parse_split_debuginfo(
1866        slot: &mut Option<SplitDebuginfo>,
1867        v: Option<&str>,
1868    ) -> bool {
1869        match v.and_then(|s| SplitDebuginfo::from_str(s).ok()) {
1870            Some(e) => *slot = Some(e),
1871            _ => return false,
1872        }
1873        true
1874    }
1875
1876    pub(crate) fn parse_split_dwarf_kind(slot: &mut SplitDwarfKind, v: Option<&str>) -> bool {
1877        match v.and_then(|s| SplitDwarfKind::from_str(s).ok()) {
1878            Some(e) => *slot = e,
1879            _ => return false,
1880        }
1881        true
1882    }
1883
1884    pub(crate) fn parse_stack_protector(slot: &mut StackProtector, v: Option<&str>) -> bool {
1885        match v.and_then(|s| StackProtector::from_str(s).ok()) {
1886            Some(ssp) => *slot = ssp,
1887            _ => return false,
1888        }
1889        true
1890    }
1891
1892    pub(crate) fn parse_branch_protection(
1893        slot: &mut Option<BranchProtection>,
1894        v: Option<&str>,
1895    ) -> bool {
1896        match v {
1897            Some(s) => {
1898                let slot = slot.get_or_insert_default();
1899                for opt in s.split(',') {
1900                    match opt {
1901                        "bti" => slot.bti = true,
1902                        "pac-ret" if slot.pac_ret.is_none() => {
1903                            slot.pac_ret = Some(PacRet { leaf: false, pc: false, key: PAuthKey::A })
1904                        }
1905                        "leaf" => match slot.pac_ret.as_mut() {
1906                            Some(pac) => pac.leaf = true,
1907                            _ => return false,
1908                        },
1909                        "b-key" => match slot.pac_ret.as_mut() {
1910                            Some(pac) => pac.key = PAuthKey::B,
1911                            _ => return false,
1912                        },
1913                        "pc" => match slot.pac_ret.as_mut() {
1914                            Some(pac) => pac.pc = true,
1915                            _ => return false,
1916                        },
1917                        "gcs" => slot.gcs = true,
1918                        _ => return false,
1919                    };
1920                }
1921            }
1922            _ => return false,
1923        }
1924        true
1925    }
1926
1927    pub(crate) fn parse_collapse_macro_debuginfo(
1928        slot: &mut CollapseMacroDebuginfo,
1929        v: Option<&str>,
1930    ) -> bool {
1931        if v.is_some() {
1932            let mut bool_arg = None;
1933            if parse_opt_bool(&mut bool_arg, v) {
1934                *slot = if bool_arg.unwrap() {
1935                    CollapseMacroDebuginfo::Yes
1936                } else {
1937                    CollapseMacroDebuginfo::No
1938                };
1939                return true;
1940            }
1941        }
1942
1943        *slot = match v {
1944            Some("external") => CollapseMacroDebuginfo::External,
1945            _ => return false,
1946        };
1947        true
1948    }
1949
1950    pub(crate) fn parse_proc_macro_execution_strategy(
1951        slot: &mut ProcMacroExecutionStrategy,
1952        v: Option<&str>,
1953    ) -> bool {
1954        *slot = match v {
1955            Some("same-thread") => ProcMacroExecutionStrategy::SameThread,
1956            Some("cross-thread") => ProcMacroExecutionStrategy::CrossThread,
1957            _ => return false,
1958        };
1959        true
1960    }
1961
1962    pub(crate) fn parse_inlining_threshold(slot: &mut InliningThreshold, v: Option<&str>) -> bool {
1963        match v {
1964            Some("always" | "yes") => {
1965                *slot = InliningThreshold::Always;
1966            }
1967            Some("never") => {
1968                *slot = InliningThreshold::Never;
1969            }
1970            Some(v) => {
1971                if let Ok(threshold) = v.parse() {
1972                    *slot = InliningThreshold::Sometimes(threshold);
1973                } else {
1974                    return false;
1975                }
1976            }
1977            None => return false,
1978        }
1979        true
1980    }
1981
1982    pub(crate) fn parse_llvm_module_flag(
1983        slot: &mut Vec<(String, u32, String)>,
1984        v: Option<&str>,
1985    ) -> bool {
1986        let elements = v.unwrap_or_default().split(':').collect::<Vec<_>>();
1987        let [key, md_type, value, behavior] = elements.as_slice() else {
1988            return false;
1989        };
1990        if *md_type != "u32" {
1991            // Currently we only support u32 metadata flags, but require the
1992            // type for forward-compatibility.
1993            return false;
1994        }
1995        let Ok(value) = value.parse::<u32>() else {
1996            return false;
1997        };
1998        let behavior = behavior.to_lowercase();
1999        let all_behaviors =
2000            ["error", "warning", "require", "override", "append", "appendunique", "max", "min"];
2001        if !all_behaviors.contains(&behavior.as_str()) {
2002            return false;
2003        }
2004
2005        slot.push((key.to_string(), value, behavior));
2006        true
2007    }
2008
2009    pub(crate) fn parse_function_return(slot: &mut FunctionReturn, v: Option<&str>) -> bool {
2010        match v {
2011            Some("keep") => *slot = FunctionReturn::Keep,
2012            Some("thunk-extern") => *slot = FunctionReturn::ThunkExtern,
2013            _ => return false,
2014        }
2015        true
2016    }
2017
2018    pub(crate) fn parse_wasm_c_abi(_slot: &mut (), v: Option<&str>) -> bool {
2019        v == Some("spec")
2020    }
2021
2022    pub(crate) fn parse_mir_include_spans(slot: &mut MirIncludeSpans, v: Option<&str>) -> bool {
2023        *slot = match v {
2024            Some("on" | "yes" | "y" | "true") | None => MirIncludeSpans::On,
2025            Some("off" | "no" | "n" | "false") => MirIncludeSpans::Off,
2026            Some("nll") => MirIncludeSpans::Nll,
2027            _ => return false,
2028        };
2029
2030        true
2031    }
2032
2033    pub(crate) fn parse_align(slot: &mut Option<Align>, v: Option<&str>) -> bool {
2034        let mut bytes = 0u64;
2035        if !parse_number(&mut bytes, v) {
2036            return false;
2037        }
2038
2039        let Ok(align) = Align::from_bytes(bytes) else {
2040            return false;
2041        };
2042
2043        *slot = Some(align);
2044
2045        true
2046    }
2047
2048    pub(crate) fn parse_assert_incr_state(
2049        slot: &mut Option<IncrementalStateAssertion>,
2050        v: Option<&str>,
2051    ) -> bool {
2052        *slot = match v {
2053            Some("loaded") => Some(IncrementalStateAssertion::Loaded),
2054            Some("not-loaded") => Some(IncrementalStateAssertion::NotLoaded),
2055            _ => return false,
2056        };
2057        true
2058    }
2059
2060    pub(crate) fn parse_rust_version(slot: &mut Option<RustcVersion>, v: Option<&str>) -> bool {
2061        let Some(v) = v else {
2062            return false;
2063        };
2064        if let Some(version) = RustcVersion::parse_str_strict(v) {
2065            *slot = Some(version);
2066            true
2067        } else {
2068            false
2069        }
2070    }
2071}
2072
2073#[rustc_lint_opt_ty]
pub struct CodegenOptions {
    pub ar: (),
    #[rustc_lint_opt_deny_field_access("use `Session::code_model` instead of this field")]
    pub code_model: Option<CodeModel>,
    pub codegen_units: Option<usize>,
    pub collapse_macro_debuginfo: CollapseMacroDebuginfo,
    pub control_flow_guard: CFGuard,
    pub debug_assertions: Option<bool>,
    pub debuginfo: DebugInfo,
    pub default_linker_libraries: bool,
    pub dlltool: Option<PathBuf>,
    #[rustc_lint_opt_deny_field_access("use `Session::dwarf_version` instead of this field")]
    pub dwarf_version: Option<u32>,
    pub embed_bitcode: bool,
    pub extra_filename: String,
    pub force_frame_pointers: FramePointer,
    #[rustc_lint_opt_deny_field_access("use `Session::must_emit_unwind_tables` instead of this field")]
    pub force_unwind_tables: Option<bool>,
    pub help: bool,
    pub incremental: Option<String>,
    pub inline_threshold: (),
    #[rustc_lint_opt_deny_field_access("use `Session::instrument_coverage` instead of this field")]
    pub instrument_coverage: InstrumentCoverage,
    pub jump_tables: bool,
    pub link_arg: (),
    pub link_args: Vec<String>,
    #[rustc_lint_opt_deny_field_access("use `Session::link_dead_code` instead of this field")]
    pub link_dead_code: Option<bool>,
    pub link_self_contained: LinkSelfContained,
    pub linker: Option<PathBuf>,
    pub linker_features: LinkerFeaturesCli,
    pub linker_flavor: Option<LinkerFlavorCli>,
    pub linker_plugin_lto: LinkerPluginLto,
    pub llvm_args: Vec<String>,
    #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
    pub lto: LtoCli,
    pub metadata: Vec<String>,
    pub no_prepopulate_passes: bool,
    pub no_redzone: Option<bool>,
    pub no_stack_check: (),
    pub no_vectorize_loops: bool,
    pub no_vectorize_slp: bool,
    pub opt_level: String,
    #[rustc_lint_opt_deny_field_access("use `Session::overflow_checks` instead of this field")]
    pub overflow_checks: Option<bool>,
    #[rustc_lint_opt_deny_field_access("use `Session::panic_strategy` instead of this field")]
    pub panic: Option<PanicStrategy>,
    pub passes: Vec<String>,
    pub prefer_dynamic: bool,
    pub profile_generate: SwitchWithOptPath,
    pub profile_use: Option<PathBuf>,
    #[rustc_lint_opt_deny_field_access("use `Session::relocation_model` instead of this field")]
    pub relocation_model: Option<RelocModel>,
    pub relro_level: Option<RelroLevel>,
    pub remark: Passes,
    pub rpath: bool,
    pub save_temps: bool,
    pub soft_float: (),
    #[rustc_lint_opt_deny_field_access("use `Session::split_debuginfo` instead of this field")]
    pub split_debuginfo: Option<SplitDebuginfo>,
    pub strip: Strip,
    pub symbol_mangling_version: Option<SymbolManglingVersion>,
    pub target_cpu: Option<String>,
    pub target_feature: String,
    pub unsafe_allow_abi_mismatch: Vec<String>,
}
#[automatically_derived]
impl ::core::clone::Clone for CodegenOptions {
    #[inline]
    fn clone(&self) -> CodegenOptions {
        CodegenOptions {
            ar: ::core::clone::Clone::clone(&self.ar),
            code_model: ::core::clone::Clone::clone(&self.code_model),
            codegen_units: ::core::clone::Clone::clone(&self.codegen_units),
            collapse_macro_debuginfo: ::core::clone::Clone::clone(&self.collapse_macro_debuginfo),
            control_flow_guard: ::core::clone::Clone::clone(&self.control_flow_guard),
            debug_assertions: ::core::clone::Clone::clone(&self.debug_assertions),
            debuginfo: ::core::clone::Clone::clone(&self.debuginfo),
            default_linker_libraries: ::core::clone::Clone::clone(&self.default_linker_libraries),
            dlltool: ::core::clone::Clone::clone(&self.dlltool),
            dwarf_version: ::core::clone::Clone::clone(&self.dwarf_version),
            embed_bitcode: ::core::clone::Clone::clone(&self.embed_bitcode),
            extra_filename: ::core::clone::Clone::clone(&self.extra_filename),
            force_frame_pointers: ::core::clone::Clone::clone(&self.force_frame_pointers),
            force_unwind_tables: ::core::clone::Clone::clone(&self.force_unwind_tables),
            help: ::core::clone::Clone::clone(&self.help),
            incremental: ::core::clone::Clone::clone(&self.incremental),
            inline_threshold: ::core::clone::Clone::clone(&self.inline_threshold),
            instrument_coverage: ::core::clone::Clone::clone(&self.instrument_coverage),
            jump_tables: ::core::clone::Clone::clone(&self.jump_tables),
            link_arg: ::core::clone::Clone::clone(&self.link_arg),
            link_args: ::core::clone::Clone::clone(&self.link_args),
            link_dead_code: ::core::clone::Clone::clone(&self.link_dead_code),
            link_self_contained: ::core::clone::Clone::clone(&self.link_self_contained),
            linker: ::core::clone::Clone::clone(&self.linker),
            linker_features: ::core::clone::Clone::clone(&self.linker_features),
            linker_flavor: ::core::clone::Clone::clone(&self.linker_flavor),
            linker_plugin_lto: ::core::clone::Clone::clone(&self.linker_plugin_lto),
            llvm_args: ::core::clone::Clone::clone(&self.llvm_args),
            lto: ::core::clone::Clone::clone(&self.lto),
            metadata: ::core::clone::Clone::clone(&self.metadata),
            no_prepopulate_passes: ::core::clone::Clone::clone(&self.no_prepopulate_passes),
            no_redzone: ::core::clone::Clone::clone(&self.no_redzone),
            no_stack_check: ::core::clone::Clone::clone(&self.no_stack_check),
            no_vectorize_loops: ::core::clone::Clone::clone(&self.no_vectorize_loops),
            no_vectorize_slp: ::core::clone::Clone::clone(&self.no_vectorize_slp),
            opt_level: ::core::clone::Clone::clone(&self.opt_level),
            overflow_checks: ::core::clone::Clone::clone(&self.overflow_checks),
            panic: ::core::clone::Clone::clone(&self.panic),
            passes: ::core::clone::Clone::clone(&self.passes),
            prefer_dynamic: ::core::clone::Clone::clone(&self.prefer_dynamic),
            profile_generate: ::core::clone::Clone::clone(&self.profile_generate),
            profile_use: ::core::clone::Clone::clone(&self.profile_use),
            relocation_model: ::core::clone::Clone::clone(&self.relocation_model),
            relro_level: ::core::clone::Clone::clone(&self.relro_level),
            remark: ::core::clone::Clone::clone(&self.remark),
            rpath: ::core::clone::Clone::clone(&self.rpath),
            save_temps: ::core::clone::Clone::clone(&self.save_temps),
            soft_float: ::core::clone::Clone::clone(&self.soft_float),
            split_debuginfo: ::core::clone::Clone::clone(&self.split_debuginfo),
            strip: ::core::clone::Clone::clone(&self.strip),
            symbol_mangling_version: ::core::clone::Clone::clone(&self.symbol_mangling_version),
            target_cpu: ::core::clone::Clone::clone(&self.target_cpu),
            target_feature: ::core::clone::Clone::clone(&self.target_feature),
            unsafe_allow_abi_mismatch: ::core::clone::Clone::clone(&self.unsafe_allow_abi_mismatch),
        }
    }
}
pub enum CodegenOptionsTargetModifiers {}
#[automatically_derived]
impl ::core::marker::StructuralPartialEq for CodegenOptionsTargetModifiers { }
#[automatically_derived]
impl ::core::cmp::PartialEq for CodegenOptionsTargetModifiers {
    #[inline]
    fn eq(&self, other: &CodegenOptionsTargetModifiers) -> bool {
        match *self {}
    }
}
#[automatically_derived]
impl ::core::cmp::Eq for CodegenOptionsTargetModifiers {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}
#[automatically_derived]
impl ::core::cmp::PartialOrd for CodegenOptionsTargetModifiers {
    #[inline]
    fn partial_cmp(&self, other: &CodegenOptionsTargetModifiers)
        -> ::core::option::Option<::core::cmp::Ordering> {
        match *self {}
    }
}
#[automatically_derived]
impl ::core::cmp::Ord for CodegenOptionsTargetModifiers {
    #[inline]
    fn cmp(&self, other: &CodegenOptionsTargetModifiers)
        -> ::core::cmp::Ordering {
        match *self {}
    }
}
#[automatically_derived]
impl ::core::fmt::Debug for CodegenOptionsTargetModifiers {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match *self {}
    }
}
#[automatically_derived]
impl ::core::marker::Copy for CodegenOptionsTargetModifiers { }
#[automatically_derived]
#[doc(hidden)]
unsafe impl ::core::clone::TrivialClone for CodegenOptionsTargetModifiers { }
#[automatically_derived]
impl ::core::clone::Clone for CodegenOptionsTargetModifiers {
    #[inline]
    fn clone(&self) -> CodegenOptionsTargetModifiers { *self }
}
const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for CodegenOptionsTargetModifiers {
            fn encode(&self, __encoder: &mut __E) { match *self {} }
        }
    };
const _: () =
    {
        impl<__D: ::rustc_span::BlobDecoder> ::rustc_serialize::Decodable<__D>
            for CodegenOptionsTargetModifiers {
            fn decode(__decoder: &mut __D) -> Self {
                {
                    ::core::panicking::panic_fmt(format_args!("`CodegenOptionsTargetModifiers` has no variants to decode"));
                }
            }
        }
    };
impl CodegenOptionsTargetModifiers {
    pub fn reparse(&self, _user_value: &str) -> ExtendedTargetModifierInfo {
        match self
            {
                #[allow(unreachable_patterns)]
                _ => {
                ::core::panicking::panic_fmt(format_args!("unknown target modifier option: {0:?}",
                        *self));
            }
        }
    }
    pub fn is_target_modifier(flag_name: &str) -> bool {
        match flag_name.replace('-', "_").as_str() { _ => false, }
    }
}
impl Default for CodegenOptions {
    fn default() -> CodegenOptions {
        CodegenOptions {
            ar: (),
            code_model: None,
            codegen_units: None,
            collapse_macro_debuginfo: CollapseMacroDebuginfo::Unspecified,
            control_flow_guard: CFGuard::Disabled,
            debug_assertions: None,
            debuginfo: DebugInfo::None,
            default_linker_libraries: false,
            dlltool: None,
            dwarf_version: None,
            embed_bitcode: true,
            extra_filename: String::new(),
            force_frame_pointers: FramePointer::MayOmit,
            force_unwind_tables: None,
            help: false,
            incremental: None,
            inline_threshold: (),
            instrument_coverage: InstrumentCoverage::No,
            jump_tables: true,
            link_arg: (),
            link_args: Vec::new(),
            link_dead_code: None,
            link_self_contained: LinkSelfContained::default(),
            linker: None,
            linker_features: LinkerFeaturesCli::default(),
            linker_flavor: None,
            linker_plugin_lto: LinkerPluginLto::Disabled,
            llvm_args: Vec::new(),
            lto: LtoCli::Unspecified,
            metadata: Vec::new(),
            no_prepopulate_passes: false,
            no_redzone: None,
            no_stack_check: (),
            no_vectorize_loops: false,
            no_vectorize_slp: false,
            opt_level: "0".to_string(),
            overflow_checks: None,
            panic: None,
            passes: Vec::new(),
            prefer_dynamic: false,
            profile_generate: SwitchWithOptPath::Disabled,
            profile_use: None,
            relocation_model: None,
            relro_level: None,
            remark: Passes::Some(Vec::new()),
            rpath: false,
            save_temps: false,
            soft_float: (),
            split_debuginfo: None,
            strip: Strip::None,
            symbol_mangling_version: None,
            target_cpu: None,
            target_feature: String::new(),
            unsafe_allow_abi_mismatch: Vec::new(),
        }
    }
}
impl CodegenOptions {
    pub fn build(early_dcx: &EarlyDiagCtxt, matches: &getopts::Matches,
        target_modifiers: &mut CollectedOptions) -> CodegenOptions {
        build_options(early_dcx, matches, target_modifiers, CG_OPTIONS, "C",
            "codegen")
    }
    fn dep_tracking_hash(&self, for_crate_hash: bool,
        error_format: ErrorOutputType) -> Hash64 {
        let mut sub_hashes = BTreeMap::new();
        {};
        {
            if (&mut sub_hashes).insert("code_model",
                        &self.code_model as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "code_model"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("collapse_macro_debuginfo",
                        &self.collapse_macro_debuginfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "collapse_macro_debuginfo"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("control_flow_guard",
                        &self.control_flow_guard as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "control_flow_guard"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debug_assertions",
                        &self.debug_assertions as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debug_assertions"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debuginfo",
                        &self.debuginfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debuginfo"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("dwarf_version",
                        &self.dwarf_version as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "dwarf_version"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("embed_bitcode",
                        &self.embed_bitcode as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "embed_bitcode"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("force_frame_pointers",
                        &self.force_frame_pointers as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "force_frame_pointers"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("force_unwind_tables",
                        &self.force_unwind_tables as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "force_unwind_tables"));
                }
            }
        };
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("instrument_coverage",
                        &self.instrument_coverage as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "instrument_coverage"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("jump_tables",
                        &self.jump_tables as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "jump_tables"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("link_dead_code",
                        &self.link_dead_code as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "link_dead_code"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("linker_plugin_lto",
                        &self.linker_plugin_lto as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "linker_plugin_lto"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("llvm_args",
                        &self.llvm_args as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "llvm_args"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("lto",
                        &self.lto as &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "lto"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("metadata",
                        &self.metadata as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "metadata"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_prepopulate_passes",
                        &self.no_prepopulate_passes as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_prepopulate_passes"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_redzone",
                        &self.no_redzone as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_redzone"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("no_vectorize_loops",
                        &self.no_vectorize_loops as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_vectorize_loops"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_vectorize_slp",
                        &self.no_vectorize_slp as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_vectorize_slp"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("opt_level",
                        &self.opt_level as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "opt_level"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("overflow_checks",
                        &self.overflow_checks as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "overflow_checks"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("panic",
                        &self.panic as &dyn dep_tracking::DepTrackingHash).is_some()
                {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "panic"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("passes",
                        &self.passes as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "passes"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("prefer_dynamic",
                        &self.prefer_dynamic as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "prefer_dynamic"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("profile_generate",
                        &self.profile_generate as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "profile_generate"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("profile_use",
                        &self.profile_use as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "profile_use"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("relocation_model",
                        &self.relocation_model as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "relocation_model"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("relro_level",
                        &self.relro_level as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "relro_level"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("split_debuginfo",
                        &self.split_debuginfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "split_debuginfo"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("symbol_mangling_version",
                        &self.symbol_mangling_version as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "symbol_mangling_version"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("target_cpu",
                        &self.target_cpu as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "target_cpu"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("target_feature",
                        &self.target_feature as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "target_feature"));
                }
            }
        };
        {};
        let mut hasher = StableHasher::new();
        dep_tracking::stable_hash(sub_hashes, &mut hasher, error_format,
            for_crate_hash);
        hasher.finish()
    }
    pub fn gather_target_modifiers(&self, _mods: &mut Vec<TargetModifier>,
        _tmod_vals: &BTreeMap<OptionsTargetModifiers, String>) {}
}
pub const CG_OPTIONS: OptionDescrs<CodegenOptions> =
    &[OptionDesc {
                    name: "ar",
                    setter: cgopts::ar,
                    type_desc: desc::parse_ignore,
                    desc: "this option has been removed",
                    removed: None.or(Some(RemovedOption::Err)),
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "code_model",
                    setter: cgopts::code_model,
                    type_desc: desc::parse_code_model,
                    desc: "choose the code model to use (`rustc --print code-models` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "codegen_units",
                    setter: cgopts::codegen_units,
                    type_desc: desc::parse_opt_number,
                    desc: "divide crate into N units to optimize in parallel",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "collapse_macro_debuginfo",
                    setter: cgopts::collapse_macro_debuginfo,
                    type_desc: desc::parse_collapse_macro_debuginfo,
                    desc: "set option to collapse debuginfo for macros",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "control_flow_guard",
                    setter: cgopts::control_flow_guard,
                    type_desc: desc::parse_cfguard,
                    desc: "use Windows Control Flow Guard (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None.or(Some(mitigation_coverage::DeniedPartialMitigationKind::ControlFlowGuard)),
                },
                OptionDesc {
                    name: "debug_assertions",
                    setter: cgopts::debug_assertions,
                    type_desc: desc::parse_opt_bool,
                    desc: "explicitly enable the `cfg(debug_assertions)` directive",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "debuginfo",
                    setter: cgopts::debuginfo,
                    type_desc: desc::parse_debuginfo,
                    desc: "debug info emission level (0-2, none, line-directives-only, \
        line-tables-only, limited, or full; default: 0)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "default_linker_libraries",
                    setter: cgopts::default_linker_libraries,
                    type_desc: desc::parse_bool,
                    desc: "allow the linker to link its default libraries (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dlltool",
                    setter: cgopts::dlltool,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "import library generation tool (ignored except when targeting windows-gnu)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dwarf_version",
                    setter: cgopts::dwarf_version,
                    type_desc: desc::parse_opt_number,
                    desc: "version of DWARF debug information to emit (default: 2 or 4, depending on platform)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "embed_bitcode",
                    setter: cgopts::embed_bitcode,
                    type_desc: desc::parse_bool,
                    desc: "emit bitcode in rlibs (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "extra_filename",
                    setter: cgopts::extra_filename,
                    type_desc: desc::parse_string,
                    desc: "extra data to put in each output filename",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "force_frame_pointers",
                    setter: cgopts::force_frame_pointers,
                    type_desc: desc::parse_frame_pointer,
                    desc: "force use of the frame pointers",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "force_unwind_tables",
                    setter: cgopts::force_unwind_tables,
                    type_desc: desc::parse_opt_bool,
                    desc: "force use of unwind tables",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "help",
                    setter: cgopts::help,
                    type_desc: desc::parse_no_value,
                    desc: "Print codegen options",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "incremental",
                    setter: cgopts::incremental,
                    type_desc: desc::parse_opt_string,
                    desc: "enable incremental compilation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_threshold",
                    setter: cgopts::inline_threshold,
                    type_desc: desc::parse_ignore,
                    desc: "this option has been removed \
        (consider using `-Cllvm-args=--inline-threshold=...`)",
                    removed: None.or(Some(RemovedOption::Err)),
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "instrument_coverage",
                    setter: cgopts::instrument_coverage,
                    type_desc: desc::parse_instrument_coverage,
                    desc: "instrument the generated code to support LLVM source-based code coverage reports \
        (note, the compiler build config must include `profiler = true`); \
        implies `-C symbol-mangling-version=v0`",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "jump_tables",
                    setter: cgopts::jump_tables,
                    type_desc: desc::parse_bool,
                    desc: "allow jump table and lookup table generation from switch case lowering (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_arg",
                    setter: cgopts::link_arg,
                    type_desc: desc::parse_string_push,
                    desc: "a single extra argument to append to the linker invocation (can be used several times)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_args",
                    setter: cgopts::link_args,
                    type_desc: desc::parse_list,
                    desc: "extra arguments to append to the linker invocation (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_dead_code",
                    setter: cgopts::link_dead_code,
                    type_desc: desc::parse_opt_bool,
                    desc: "try to generate and link dead code (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_self_contained",
                    setter: cgopts::link_self_contained,
                    type_desc: desc::parse_link_self_contained,
                    desc: "control whether to link Rust provided C objects/libraries or rely \
        on a C toolchain or linker installed in the system",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "linker",
                    setter: cgopts::linker,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "system linker to link outputs with",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "linker_features",
                    setter: cgopts::linker_features,
                    type_desc: desc::parse_linker_features,
                    desc: "a comma-separated list of linker features to enable (+) or disable (-): `lld`",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "linker_flavor",
                    setter: cgopts::linker_flavor,
                    type_desc: desc::parse_linker_flavor,
                    desc: "linker flavor",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "linker_plugin_lto",
                    setter: cgopts::linker_plugin_lto,
                    type_desc: desc::parse_linker_plugin_lto,
                    desc: "generate build artifacts that are compatible with linker-based LTO",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "llvm_args",
                    setter: cgopts::llvm_args,
                    type_desc: desc::parse_list,
                    desc: "a list of arguments to pass to LLVM (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "lto",
                    setter: cgopts::lto,
                    type_desc: desc::parse_lto,
                    desc: "perform LLVM link-time optimizations",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "metadata",
                    setter: cgopts::metadata,
                    type_desc: desc::parse_list,
                    desc: "metadata to mangle symbol names with",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_prepopulate_passes",
                    setter: cgopts::no_prepopulate_passes,
                    type_desc: desc::parse_no_value,
                    desc: "give an empty list of passes to the pass manager",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_redzone",
                    setter: cgopts::no_redzone,
                    type_desc: desc::parse_opt_bool,
                    desc: "disable the use of the redzone",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_stack_check",
                    setter: cgopts::no_stack_check,
                    type_desc: desc::parse_ignore,
                    desc: "this option has been removed",
                    removed: None.or(Some(RemovedOption::Err)),
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_vectorize_loops",
                    setter: cgopts::no_vectorize_loops,
                    type_desc: desc::parse_no_value,
                    desc: "disable loop vectorization optimization passes",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_vectorize_slp",
                    setter: cgopts::no_vectorize_slp,
                    type_desc: desc::parse_no_value,
                    desc: "disable LLVM's SLP vectorization pass",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "opt_level",
                    setter: cgopts::opt_level,
                    type_desc: desc::parse_string,
                    desc: "optimization level (0-3, s, or z; default: 0)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "overflow_checks",
                    setter: cgopts::overflow_checks,
                    type_desc: desc::parse_opt_bool,
                    desc: "use overflow checks for integer arithmetic",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "panic",
                    setter: cgopts::panic,
                    type_desc: desc::parse_opt_panic_strategy,
                    desc: "panic strategy to compile crate with",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "passes",
                    setter: cgopts::passes,
                    type_desc: desc::parse_list,
                    desc: "a list of extra LLVM passes to run (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "prefer_dynamic",
                    setter: cgopts::prefer_dynamic,
                    type_desc: desc::parse_bool,
                    desc: "prefer dynamic linking to static linking (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "profile_generate",
                    setter: cgopts::profile_generate,
                    type_desc: desc::parse_switch_with_opt_path,
                    desc: "compile the program with profiling instrumentation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "profile_use",
                    setter: cgopts::profile_use,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "use the given `.profdata` file for profile-guided optimization",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "relocation_model",
                    setter: cgopts::relocation_model,
                    type_desc: desc::parse_relocation_model,
                    desc: "control generation of position-independent code (PIC) \
        (`rustc --print relocation-models` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "relro_level",
                    setter: cgopts::relro_level,
                    type_desc: desc::parse_relro_level,
                    desc: "choose which RELRO level to use",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "remark",
                    setter: cgopts::remark,
                    type_desc: desc::parse_passes,
                    desc: "output remarks for these optimization passes (space separated, or \"all\")",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "rpath",
                    setter: cgopts::rpath,
                    type_desc: desc::parse_bool,
                    desc: "set rpath values in libs/exes (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "save_temps",
                    setter: cgopts::save_temps,
                    type_desc: desc::parse_bool,
                    desc: "save all temporary output files during compilation (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "soft_float",
                    setter: cgopts::soft_float,
                    type_desc: desc::parse_ignore,
                    desc: "this option has been removed \
        (use a corresponding *eabi target instead)",
                    removed: None.or(Some(RemovedOption::Err)),
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "split_debuginfo",
                    setter: cgopts::split_debuginfo,
                    type_desc: desc::parse_split_debuginfo,
                    desc: "how to handle split-debuginfo, a platform-specific option",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "strip",
                    setter: cgopts::strip,
                    type_desc: desc::parse_strip,
                    desc: "tell the linker which information to strip (`none` (default), `debuginfo` or `symbols`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "symbol_mangling_version",
                    setter: cgopts::symbol_mangling_version,
                    type_desc: desc::parse_symbol_mangling_version,
                    desc: "which mangling version to use for symbol names ('legacy', 'v0' (default), or 'hashed')",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "target_cpu",
                    setter: cgopts::target_cpu,
                    type_desc: desc::parse_opt_string,
                    desc: "select target processor (`rustc --print target-cpus` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "target_feature",
                    setter: cgopts::target_feature,
                    type_desc: desc::parse_target_feature,
                    desc: "target specific attributes. (`rustc --print target-features` for details). \
        This feature is unsafe.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "unsafe_allow_abi_mismatch",
                    setter: cgopts::unsafe_allow_abi_mismatch,
                    type_desc: desc::parse_comma_list,
                    desc: "Allow incompatible target modifiers in dependency crates (comma separated list)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                }];
mod cgopts {
    pub(super) fn ar(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_ignore(&mut cg.ar, v)
    }
    pub(super) fn code_model(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_code_model(&mut cg.code_model, v)
    }
    pub(super) fn codegen_units(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.codegen_units, v)
    }
    pub(super) fn collapse_macro_debuginfo(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_collapse_macro_debuginfo(&mut cg.collapse_macro_debuginfo,
            v)
    }
    pub(super) fn control_flow_guard(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_cfguard(&mut cg.control_flow_guard, v)
    }
    pub(super) fn debug_assertions(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.debug_assertions, v)
    }
    pub(super) fn debuginfo(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_debuginfo(&mut cg.debuginfo, v)
    }
    pub(super) fn default_linker_libraries(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.default_linker_libraries, v)
    }
    pub(super) fn dlltool(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.dlltool, v)
    }
    pub(super) fn dwarf_version(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.dwarf_version, v)
    }
    pub(super) fn embed_bitcode(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.embed_bitcode, v)
    }
    pub(super) fn extra_filename(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.extra_filename, v)
    }
    pub(super) fn force_frame_pointers(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_frame_pointer(&mut cg.force_frame_pointers, v)
    }
    pub(super) fn force_unwind_tables(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.force_unwind_tables, v)
    }
    pub(super) fn help(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.help, v)
    }
    pub(super) fn incremental(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.incremental, v)
    }
    pub(super) fn inline_threshold(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_ignore(&mut cg.inline_threshold, v)
    }
    pub(super) fn instrument_coverage(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_instrument_coverage(&mut cg.instrument_coverage,
            v)
    }
    pub(super) fn jump_tables(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.jump_tables, v)
    }
    pub(super) fn link_arg(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string_push(&mut cg.link_args, v)
    }
    pub(super) fn link_args(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.link_args, v)
    }
    pub(super) fn link_dead_code(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.link_dead_code, v)
    }
    pub(super) fn link_self_contained(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_link_self_contained(&mut cg.link_self_contained,
            v)
    }
    pub(super) fn linker(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.linker, v)
    }
    pub(super) fn linker_features(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_linker_features(&mut cg.linker_features, v)
    }
    pub(super) fn linker_flavor(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_linker_flavor(&mut cg.linker_flavor, v)
    }
    pub(super) fn linker_plugin_lto(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_linker_plugin_lto(&mut cg.linker_plugin_lto, v)
    }
    pub(super) fn llvm_args(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.llvm_args, v)
    }
    pub(super) fn lto(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_lto(&mut cg.lto, v)
    }
    pub(super) fn metadata(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.metadata, v)
    }
    pub(super) fn no_prepopulate_passes(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_prepopulate_passes, v)
    }
    pub(super) fn no_redzone(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.no_redzone, v)
    }
    pub(super) fn no_stack_check(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_ignore(&mut cg.no_stack_check, v)
    }
    pub(super) fn no_vectorize_loops(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_vectorize_loops, v)
    }
    pub(super) fn no_vectorize_slp(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_vectorize_slp, v)
    }
    pub(super) fn opt_level(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.opt_level, v)
    }
    pub(super) fn overflow_checks(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.overflow_checks, v)
    }
    pub(super) fn panic(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_panic_strategy(&mut cg.panic, v)
    }
    pub(super) fn passes(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.passes, v)
    }
    pub(super) fn prefer_dynamic(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.prefer_dynamic, v)
    }
    pub(super) fn profile_generate(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_switch_with_opt_path(&mut cg.profile_generate, v)
    }
    pub(super) fn profile_use(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.profile_use, v)
    }
    pub(super) fn relocation_model(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_relocation_model(&mut cg.relocation_model, v)
    }
    pub(super) fn relro_level(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_relro_level(&mut cg.relro_level, v)
    }
    pub(super) fn remark(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_passes(&mut cg.remark, v)
    }
    pub(super) fn rpath(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.rpath, v)
    }
    pub(super) fn save_temps(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.save_temps, v)
    }
    pub(super) fn soft_float(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_ignore(&mut cg.soft_float, v)
    }
    pub(super) fn split_debuginfo(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_split_debuginfo(&mut cg.split_debuginfo, v)
    }
    pub(super) fn strip(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_strip(&mut cg.strip, v)
    }
    pub(super) fn symbol_mangling_version(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_symbol_mangling_version(&mut cg.symbol_mangling_version,
            v)
    }
    pub(super) fn target_cpu(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.target_cpu, v)
    }
    pub(super) fn target_feature(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_target_feature(&mut cg.target_feature, v)
    }
    pub(super) fn unsafe_allow_abi_mismatch(cg: &mut super::CodegenOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_comma_list(&mut cg.unsafe_allow_abi_mismatch, v)
    }
}options! {
2074    CodegenOptions, CodegenOptionsTargetModifiers, CG_OPTIONS, cgopts, "C", "codegen",
2075
2076    // If you add a new option, please update:
2077    // - compiler/rustc_interface/src/tests.rs
2078    // - src/doc/rustc/src/codegen-options/index.md
2079
2080    // tidy-alphabetical-start
2081    ar: () = ((), parse_ignore, [UNTRACKED],
2082        "this option has been removed",
2083        removed: Err),
2084    #[rustc_lint_opt_deny_field_access("use `Session::code_model` instead of this field")]
2085    code_model: Option<CodeModel> = (None, parse_code_model, [TRACKED],
2086        "choose the code model to use (`rustc --print code-models` for details)"),
2087    codegen_units: Option<usize> = (None, parse_opt_number, [UNTRACKED],
2088        "divide crate into N units to optimize in parallel"),
2089    collapse_macro_debuginfo: CollapseMacroDebuginfo = (CollapseMacroDebuginfo::Unspecified,
2090        parse_collapse_macro_debuginfo, [TRACKED],
2091        "set option to collapse debuginfo for macros"),
2092    control_flow_guard: CFGuard = (CFGuard::Disabled, parse_cfguard, [TRACKED] { MITIGATION: ControlFlowGuard },
2093        "use Windows Control Flow Guard (default: no)"),
2094    debug_assertions: Option<bool> = (None, parse_opt_bool, [TRACKED],
2095        "explicitly enable the `cfg(debug_assertions)` directive"),
2096    debuginfo: DebugInfo = (DebugInfo::None, parse_debuginfo, [TRACKED],
2097        "debug info emission level (0-2, none, line-directives-only, \
2098        line-tables-only, limited, or full; default: 0)"),
2099    default_linker_libraries: bool = (false, parse_bool, [UNTRACKED],
2100        "allow the linker to link its default libraries (default: no)"),
2101    dlltool: Option<PathBuf> = (None, parse_opt_pathbuf, [UNTRACKED],
2102        "import library generation tool (ignored except when targeting windows-gnu)"),
2103    #[rustc_lint_opt_deny_field_access("use `Session::dwarf_version` instead of this field")]
2104    dwarf_version: Option<u32> = (None, parse_opt_number, [TRACKED],
2105        "version of DWARF debug information to emit (default: 2 or 4, depending on platform)"),
2106    embed_bitcode: bool = (true, parse_bool, [TRACKED],
2107        "emit bitcode in rlibs (default: yes)"),
2108    extra_filename: String = (String::new(), parse_string, [UNTRACKED],
2109        "extra data to put in each output filename"),
2110    force_frame_pointers: FramePointer = (FramePointer::MayOmit, parse_frame_pointer, [TRACKED],
2111        "force use of the frame pointers"),
2112    #[rustc_lint_opt_deny_field_access("use `Session::must_emit_unwind_tables` instead of this field")]
2113    force_unwind_tables: Option<bool> = (None, parse_opt_bool, [TRACKED],
2114        "force use of unwind tables"),
2115    help: bool = (false, parse_no_value, [UNTRACKED], "Print codegen options"),
2116    incremental: Option<String> = (None, parse_opt_string, [UNTRACKED],
2117        "enable incremental compilation"),
2118    inline_threshold: () = ((), parse_ignore, [UNTRACKED],
2119        "this option has been removed \
2120        (consider using `-Cllvm-args=--inline-threshold=...`)",
2121        removed: Err),
2122    #[rustc_lint_opt_deny_field_access("use `Session::instrument_coverage` instead of this field")]
2123    instrument_coverage: InstrumentCoverage = (InstrumentCoverage::No, parse_instrument_coverage, [TRACKED],
2124        "instrument the generated code to support LLVM source-based code coverage reports \
2125        (note, the compiler build config must include `profiler = true`); \
2126        implies `-C symbol-mangling-version=v0`"),
2127    jump_tables: bool = (true, parse_bool, [TRACKED],
2128        "allow jump table and lookup table generation from switch case lowering (default: yes)"),
2129    link_arg: (/* redirected to link_args */) = ((), parse_string_push, [UNTRACKED],
2130        "a single extra argument to append to the linker invocation (can be used several times)"),
2131    link_args: Vec<String> = (Vec::new(), parse_list, [UNTRACKED],
2132        "extra arguments to append to the linker invocation (space separated)"),
2133    #[rustc_lint_opt_deny_field_access("use `Session::link_dead_code` instead of this field")]
2134    link_dead_code: Option<bool> = (None, parse_opt_bool, [TRACKED],
2135        "try to generate and link dead code (default: no)"),
2136    link_self_contained: LinkSelfContained = (LinkSelfContained::default(), parse_link_self_contained, [UNTRACKED],
2137        "control whether to link Rust provided C objects/libraries or rely \
2138        on a C toolchain or linker installed in the system"),
2139    linker: Option<PathBuf> = (None, parse_opt_pathbuf, [UNTRACKED],
2140        "system linker to link outputs with"),
2141    linker_features: LinkerFeaturesCli = (LinkerFeaturesCli::default(), parse_linker_features, [UNTRACKED],
2142        "a comma-separated list of linker features to enable (+) or disable (-): `lld`"),
2143    linker_flavor: Option<LinkerFlavorCli> = (None, parse_linker_flavor, [UNTRACKED],
2144        "linker flavor"),
2145    linker_plugin_lto: LinkerPluginLto = (LinkerPluginLto::Disabled,
2146        parse_linker_plugin_lto, [TRACKED],
2147        "generate build artifacts that are compatible with linker-based LTO"),
2148    llvm_args: Vec<String> = (Vec::new(), parse_list, [TRACKED],
2149        "a list of arguments to pass to LLVM (space separated)"),
2150    #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
2151    lto: LtoCli = (LtoCli::Unspecified, parse_lto, [TRACKED],
2152        "perform LLVM link-time optimizations"),
2153    metadata: Vec<String> = (Vec::new(), parse_list, [TRACKED],
2154        "metadata to mangle symbol names with"),
2155    no_prepopulate_passes: bool = (false, parse_no_value, [TRACKED],
2156        "give an empty list of passes to the pass manager"),
2157    no_redzone: Option<bool> = (None, parse_opt_bool, [TRACKED],
2158        "disable the use of the redzone"),
2159    no_stack_check: () = ((), parse_ignore, [UNTRACKED],
2160        "this option has been removed",
2161        removed: Err),
2162    no_vectorize_loops: bool = (false, parse_no_value, [TRACKED],
2163        "disable loop vectorization optimization passes"),
2164    no_vectorize_slp: bool = (false, parse_no_value, [TRACKED],
2165        "disable LLVM's SLP vectorization pass"),
2166    opt_level: String = ("0".to_string(), parse_string, [TRACKED],
2167        "optimization level (0-3, s, or z; default: 0)"),
2168    #[rustc_lint_opt_deny_field_access("use `Session::overflow_checks` instead of this field")]
2169    overflow_checks: Option<bool> = (None, parse_opt_bool, [TRACKED],
2170        "use overflow checks for integer arithmetic"),
2171    #[rustc_lint_opt_deny_field_access("use `Session::panic_strategy` instead of this field")]
2172    panic: Option<PanicStrategy> = (None, parse_opt_panic_strategy, [TRACKED],
2173        "panic strategy to compile crate with"),
2174    passes: Vec<String> = (Vec::new(), parse_list, [TRACKED],
2175        "a list of extra LLVM passes to run (space separated)"),
2176    prefer_dynamic: bool = (false, parse_bool, [TRACKED],
2177        "prefer dynamic linking to static linking (default: no)"),
2178    profile_generate: SwitchWithOptPath = (SwitchWithOptPath::Disabled,
2179        parse_switch_with_opt_path, [TRACKED],
2180        "compile the program with profiling instrumentation"),
2181    profile_use: Option<PathBuf> = (None, parse_opt_pathbuf, [TRACKED],
2182        "use the given `.profdata` file for profile-guided optimization"),
2183    #[rustc_lint_opt_deny_field_access("use `Session::relocation_model` instead of this field")]
2184    relocation_model: Option<RelocModel> = (None, parse_relocation_model, [TRACKED],
2185        "control generation of position-independent code (PIC) \
2186        (`rustc --print relocation-models` for details)"),
2187    relro_level: Option<RelroLevel> = (None, parse_relro_level, [TRACKED],
2188        "choose which RELRO level to use"),
2189    remark: Passes = (Passes::Some(Vec::new()), parse_passes, [UNTRACKED],
2190        "output remarks for these optimization passes (space separated, or \"all\")"),
2191    rpath: bool = (false, parse_bool, [UNTRACKED],
2192        "set rpath values in libs/exes (default: no)"),
2193    save_temps: bool = (false, parse_bool, [UNTRACKED],
2194        "save all temporary output files during compilation (default: no)"),
2195    soft_float: () = ((), parse_ignore, [UNTRACKED],
2196        "this option has been removed \
2197        (use a corresponding *eabi target instead)",
2198        removed: Err),
2199    #[rustc_lint_opt_deny_field_access("use `Session::split_debuginfo` instead of this field")]
2200    split_debuginfo: Option<SplitDebuginfo> = (None, parse_split_debuginfo, [TRACKED],
2201        "how to handle split-debuginfo, a platform-specific option"),
2202    strip: Strip = (Strip::None, parse_strip, [UNTRACKED],
2203        "tell the linker which information to strip (`none` (default), `debuginfo` or `symbols`)"),
2204    symbol_mangling_version: Option<SymbolManglingVersion> = (None,
2205        parse_symbol_mangling_version, [TRACKED],
2206        "which mangling version to use for symbol names ('legacy', 'v0' (default), or 'hashed')"),
2207    target_cpu: Option<String> = (None, parse_opt_string, [TRACKED],
2208        "select target processor (`rustc --print target-cpus` for details)"),
2209    target_feature: String = (String::new(), parse_target_feature, [TRACKED],
2210        "target specific attributes. (`rustc --print target-features` for details). \
2211        This feature is unsafe."),
2212    unsafe_allow_abi_mismatch: Vec<String> = (Vec::new(), parse_comma_list, [UNTRACKED],
2213        "Allow incompatible target modifiers in dependency crates (comma separated list)"),
2214    // tidy-alphabetical-end
2215
2216    // If you add a new option, please update:
2217    // - compiler/rustc_interface/src/tests.rs
2218    // - src/doc/rustc/src/codegen-options/index.md
2219}
2220
2221#[rustc_lint_opt_ty]
pub struct UnstableOptions {
    pub allow_features: Option<Vec<String>>,
    pub allow_partial_mitigations: (),
    pub always_encode_mir: bool,
    pub annotate_moves: AnnotateMoves,
    pub assert_incr_state: Option<IncrementalStateAssertion>,
    pub assume_incomplete_release: bool,
    pub assumptions_on_binders: bool,
    pub autodiff: Vec<crate::config::AutoDiff>,
    #[rustc_lint_opt_deny_field_access("use `Session::binary_dep_depinfo` instead of this field")]
    pub binary_dep_depinfo: bool,
    pub box_noalias: bool,
    pub branch_protection: Option<BranchProtection>,
    pub build_sdylib_interface: bool,
    pub cache_proc_macros: bool,
    pub cf_protection: CFProtection,
    pub check_cfg_all_expected: bool,
    pub checksum_hash_algorithm: Option<SourceFileHashAlgorithm>,
    pub codegen_backend: Option<String>,
    pub codegen_emit_retag: Option<CodegenRetagOptions>,
    pub codegen_source_order: bool,
    pub contract_checks: Option<bool>,
    pub coverage_options: CoverageOptions,
    pub crate_attr: Vec<String>,
    pub cross_crate_inline_threshold: InliningThreshold,
    pub debug_info_type_line_numbers: bool,
    pub debuginfo_compression: DebugInfoCompression,
    pub debuginfo_for_profiling: bool,
    pub deduplicate_diagnostics: bool,
    pub default_visibility: Option<SymbolVisibility>,
    pub deny_partial_mitigations: (),
    pub dep_info_omit_d_target: bool,
    pub direct_access_external_data: Option<bool>,
    pub disable_fast_paths: bool,
    pub disable_incr_comp_backend_caching: bool,
    pub dual_proc_macros: bool,
    pub dump_dep_graph: bool,
    pub dump_mir: Option<String>,
    pub dump_mir_dataflow: bool,
    pub dump_mir_dir: String,
    pub dump_mir_exclude_alloc_bytes: bool,
    pub dump_mir_exclude_pass_number: bool,
    pub dump_mir_graphviz: bool,
    pub dump_mono_stats: SwitchWithOptPath,
    pub dump_mono_stats_format: DumpMonoStatsFormat,
    #[rustc_lint_opt_deny_field_access("use `Session::dwarf_version` instead of this field")]
    pub dwarf_version: Option<u32>,
    pub dylib_lto: bool,
    pub eagerly_emit_delayed_bugs: bool,
    pub ehcont_guard: bool,
    pub embed_metadata: bool,
    pub embed_source: bool,
    pub emit_stack_sizes: bool,
    pub enforce_type_length_limit: bool,
    pub experimental_default_bounds: bool,
    pub export_executable_symbols: bool,
    pub external_clangrt: bool,
    pub extra_const_ub_checks: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::fewer_names` instead of this field")]
    pub fewer_names: Option<bool>,
    pub fixed_x18: bool,
    pub flatten_format_args: bool,
    pub fmt_debug: FmtDebug,
    pub force_unstable_if_unmarked: bool,
    pub function_return: FunctionReturn,
    pub function_sections: Option<bool>,
    pub future_incompat_test: bool,
    pub graphviz_dark_mode: bool,
    pub graphviz_font: String,
    pub has_thread_local: Option<bool>,
    pub help: bool,
    pub higher_ranked_assumptions: bool,
    pub hint_mostly_unused: bool,
    pub human_readable_cgu_names: bool,
    pub identify_regions: bool,
    pub ignore_directory_in_diagnostics_source_blocks: Vec<String>,
    pub incremental_ignore_spans: bool,
    pub incremental_info: bool,
    pub incremental_verify_ich: bool,
    pub indirect_branch_cs_prefix: bool,
    pub inline_llvm: bool,
    pub inline_mir: Option<bool>,
    pub inline_mir_forwarder_threshold: Option<usize>,
    pub inline_mir_hint_threshold: Option<usize>,
    pub inline_mir_preserve_debug: Option<bool>,
    pub inline_mir_threshold: Option<usize>,
    pub input_stats: bool,
    pub instrument_mcount: bool,
    pub instrument_xray: Option<InstrumentXRay>,
    pub internal_testing_features: bool,
    pub large_data_threshold: Option<u64>,
    pub layout_seed: Option<u64>,
    pub link_directives: bool,
    pub link_native_libraries: bool,
    pub link_only: bool,
    pub lint_llvm_ir: bool,
    pub lint_mir: bool,
    pub lint_rust_version: Option<RustcVersion>,
    pub llvm_module_flag: Vec<(String, u32, String)>,
    pub llvm_plugins: Vec<String>,
    pub llvm_time_trace: bool,
    pub llvm_writable: bool,
    pub location_detail: LocationDetail,
    pub ls: Vec<String>,
    pub macro_backtrace: bool,
    pub macro_stats: bool,
    pub maximal_hir_to_mir_coverage: bool,
    pub merge_functions: Option<MergeFunctions>,
    pub meta_stats: bool,
    pub metrics_dir: Option<PathBuf>,
    pub min_function_alignment: Option<Align>,
    pub min_recursion_limit: Option<usize>,
    pub mir_enable_passes: Vec<(String, bool)>,
    pub mir_include_spans: MirIncludeSpans,
    pub mir_opt_bisect_limit: Option<usize>,
    #[rustc_lint_opt_deny_field_access("use `Session::mir_opt_level` instead of this field")]
    pub mir_opt_level: Option<usize>,
    pub mir_preserve_ub: bool,
    pub mir_strip_debuginfo: MirStripDebugInfo,
    pub move_size_limit: Option<usize>,
    pub mutable_noalias: bool,
    pub namespaced_crates: bool,
    pub next_solver: NextSolverConfig,
    pub nll_facts: bool,
    pub nll_facts_dir: String,
    pub no_analysis: bool,
    pub no_codegen: bool,
    pub no_generate_arange_section: bool,
    pub no_implied_bounds_compat: bool,
    pub no_leak_check: bool,
    pub no_link: bool,
    pub no_parallel_backend: bool,
    pub no_profiler_runtime: bool,
    pub no_steal_thir: bool,
    pub no_trait_vptr: bool,
    pub no_unique_section_names: bool,
    pub normalize_docs: bool,
    pub offload: Vec<crate::config::Offload>,
    pub on_broken_pipe: OnBrokenPipe,
    pub osx_rpath_install_name: bool,
    pub packed_bundled_libs: bool,
    pub packed_stack: bool,
    pub panic_abort_tests: bool,
    pub panic_in_drop: PanicStrategy,
    pub parse_crate_root_only: bool,
    pub patchable_function_entry: PatchableFunctionEntry,
    pub plt: Option<bool>,
    pub polonius: Polonius,
    pub pre_link_arg: (),
    pub pre_link_args: Vec<String>,
    pub precise_enum_drop_elaboration: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::print_codegen_stats` instead of this field")]
    pub print_codegen_stats: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::print_llvm_stats_json` instead of this field")]
    pub print_codegen_stats_json: Option<String>,
    pub print_llvm_passes: bool,
    pub print_mono_items: bool,
    pub print_type_sizes: bool,
    pub proc_macro_backtrace: bool,
    pub proc_macro_execution_strategy: ProcMacroExecutionStrategy,
    pub profile_closures: bool,
    pub profile_sample_use: Option<PathBuf>,
    pub profiler_runtime: String,
    pub query_dep_graph: bool,
    pub randomize_layout: bool,
    pub reg_struct_return: bool,
    pub regparm: Option<u32>,
    pub relax_elf_relocations: Option<bool>,
    pub remap_cwd_prefix: Option<PathBuf>,
    pub remark_dir: Option<PathBuf>,
    pub retpoline: bool,
    pub retpoline_external_thunk: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::sanitizers()` instead of this field")]
    pub sanitizer: SanitizerSet,
    pub sanitizer_cfi_canonical_jump_tables: Option<bool>,
    pub sanitizer_cfi_generalize_pointers: Option<bool>,
    pub sanitizer_cfi_normalize_integers: Option<bool>,
    pub sanitizer_dataflow_abilist: Vec<String>,
    pub sanitizer_kcfi_arity: Option<bool>,
    pub sanitizer_memory_track_origins: usize,
    pub sanitizer_recover: SanitizerSet,
    pub saturating_float_casts: Option<bool>,
    pub self_profile: SwitchWithOptPath,
    pub self_profile_counter: String,
    #[doc =
    r" keep this in sync with the event filter names in librustc_data_structures/profiling.rs"]
    pub self_profile_events: Option<Vec<String>>,
    pub share_generics: Option<bool>,
    pub shell_argfiles: bool,
    pub simulate_remapped_rust_src_base: Option<PathBuf>,
    pub small_data_threshold: Option<usize>,
    pub span_debug: bool,
    #[doc = r" o/w tests have closure@path"]
    pub span_free_formats: bool,
    pub split_dwarf_inlining: bool,
    pub split_dwarf_kind: SplitDwarfKind,
    pub split_dwarf_out_dir: Option<PathBuf>,
    pub split_lto_unit: Option<bool>,
    pub src_hash_algorithm: Option<SourceFileHashAlgorithm>,
    #[rustc_lint_opt_deny_field_access("use `Session::stack_protector` instead of this field")]
    pub stack_protector: StackProtector,
    pub staticlib_allow_rdylib_deps: bool,
    pub staticlib_hide_internal_symbols: bool,
    pub staticlib_rename_internal_symbols: bool,
    pub staticlib_prefer_dynamic: bool,
    pub strict_init_checks: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::teach` instead of this field")]
    pub teach: bool,
    pub temps_dir: Option<String>,
    pub terminal_urls: TerminalUrl,
    #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
    pub thinlto: Option<bool>,
    #[doc = r" We default to None here since we want to behave like"]
    #[doc = r" a sequential compiler for now. This'll likely be adjusted"]
    #[doc = r" in the future. Note that -Zthreads=0 is the way to get"]
    #[doc = r" the num_cpus behavior."]
    #[rustc_lint_opt_deny_field_access("use `Session::threads` instead of this field")]
    pub threads: Option<usize>,
    pub time_llvm_passes: bool,
    pub time_passes: bool,
    pub time_passes_format: TimePassesFormat,
    pub tiny_const_eval_limit: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::tls_model` instead of this field")]
    pub tls_model: Option<TlsModel>,
    pub trace_macros: bool,
    pub track_diagnostics: bool,
    pub translate_remapped_path_to_local_path: bool,
    pub trap_unreachable: Option<bool>,
    pub treat_err_as_bug: Option<NonZero<usize>>,
    pub trim_diagnostic_paths: bool,
    pub tune_cpu: Option<String>,
    #[rustc_lint_opt_deny_field_access("use `TyCtxt::use_typing_mode_post_typeck` instead of this field")]
    pub typing_mode_post_typeck_until_borrowck: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::ub_checks` instead of this field")]
    pub ub_checks: Option<bool>,
    pub ui_testing: bool,
    pub uninit_const_chunk_threshold: usize,
    pub unleash_the_miri_inside_of_you: bool,
    pub unpretty: Option<String>,
    pub unsound_mir_opts: bool,
    #[doc =
    r" This name is kind of confusing: Most unstable options enable something themselves, while"]
    #[doc = r#" this just allows "normal" options to be feature-gated."#]
    #[doc = r""]
    #[doc =
    r" The main check for `-Zunstable-options` takes place separately from the"]
    #[doc =
    r" usual parsing of `-Z` options (see [`crate::config::nightly_options`]),"]
    #[doc =
    r" so this boolean value is mostly used for enabling unstable _values_ of"]
    #[doc =
    r" stable options. That separate check doesn't handle boolean values, so"]
    #[doc = r" to avoid an inconsistent state we also forbid them here."]
    #[rustc_lint_opt_deny_field_access("use `Session::unstable_options` instead of this field")]
    pub unstable_options: bool,
    pub use_ctors_section: Option<bool>,
    pub use_sync_unwind: Option<bool>,
    pub validate_mir: bool,
    pub verbose_asm: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::verbose_internals` instead of this field")]
    pub verbose_internals: bool,
    #[rustc_lint_opt_deny_field_access("use `Session::verify_llvm_ir` instead of this field")]
    pub verify_llvm_ir: bool,
    pub virtual_function_elimination: bool,
    pub wasi_exec_model: Option<WasiExecModel>,
    pub wasm_c_abi: (),
    pub write_long_types_to_disk: bool,
}
#[automatically_derived]
impl ::core::clone::Clone for UnstableOptions {
    #[inline]
    fn clone(&self) -> UnstableOptions {
        UnstableOptions {
            allow_features: ::core::clone::Clone::clone(&self.allow_features),
            allow_partial_mitigations: ::core::clone::Clone::clone(&self.allow_partial_mitigations),
            always_encode_mir: ::core::clone::Clone::clone(&self.always_encode_mir),
            annotate_moves: ::core::clone::Clone::clone(&self.annotate_moves),
            assert_incr_state: ::core::clone::Clone::clone(&self.assert_incr_state),
            assume_incomplete_release: ::core::clone::Clone::clone(&self.assume_incomplete_release),
            assumptions_on_binders: ::core::clone::Clone::clone(&self.assumptions_on_binders),
            autodiff: ::core::clone::Clone::clone(&self.autodiff),
            binary_dep_depinfo: ::core::clone::Clone::clone(&self.binary_dep_depinfo),
            box_noalias: ::core::clone::Clone::clone(&self.box_noalias),
            branch_protection: ::core::clone::Clone::clone(&self.branch_protection),
            build_sdylib_interface: ::core::clone::Clone::clone(&self.build_sdylib_interface),
            cache_proc_macros: ::core::clone::Clone::clone(&self.cache_proc_macros),
            cf_protection: ::core::clone::Clone::clone(&self.cf_protection),
            check_cfg_all_expected: ::core::clone::Clone::clone(&self.check_cfg_all_expected),
            checksum_hash_algorithm: ::core::clone::Clone::clone(&self.checksum_hash_algorithm),
            codegen_backend: ::core::clone::Clone::clone(&self.codegen_backend),
            codegen_emit_retag: ::core::clone::Clone::clone(&self.codegen_emit_retag),
            codegen_source_order: ::core::clone::Clone::clone(&self.codegen_source_order),
            contract_checks: ::core::clone::Clone::clone(&self.contract_checks),
            coverage_options: ::core::clone::Clone::clone(&self.coverage_options),
            crate_attr: ::core::clone::Clone::clone(&self.crate_attr),
            cross_crate_inline_threshold: ::core::clone::Clone::clone(&self.cross_crate_inline_threshold),
            debug_info_type_line_numbers: ::core::clone::Clone::clone(&self.debug_info_type_line_numbers),
            debuginfo_compression: ::core::clone::Clone::clone(&self.debuginfo_compression),
            debuginfo_for_profiling: ::core::clone::Clone::clone(&self.debuginfo_for_profiling),
            deduplicate_diagnostics: ::core::clone::Clone::clone(&self.deduplicate_diagnostics),
            default_visibility: ::core::clone::Clone::clone(&self.default_visibility),
            deny_partial_mitigations: ::core::clone::Clone::clone(&self.deny_partial_mitigations),
            dep_info_omit_d_target: ::core::clone::Clone::clone(&self.dep_info_omit_d_target),
            direct_access_external_data: ::core::clone::Clone::clone(&self.direct_access_external_data),
            disable_fast_paths: ::core::clone::Clone::clone(&self.disable_fast_paths),
            disable_incr_comp_backend_caching: ::core::clone::Clone::clone(&self.disable_incr_comp_backend_caching),
            dual_proc_macros: ::core::clone::Clone::clone(&self.dual_proc_macros),
            dump_dep_graph: ::core::clone::Clone::clone(&self.dump_dep_graph),
            dump_mir: ::core::clone::Clone::clone(&self.dump_mir),
            dump_mir_dataflow: ::core::clone::Clone::clone(&self.dump_mir_dataflow),
            dump_mir_dir: ::core::clone::Clone::clone(&self.dump_mir_dir),
            dump_mir_exclude_alloc_bytes: ::core::clone::Clone::clone(&self.dump_mir_exclude_alloc_bytes),
            dump_mir_exclude_pass_number: ::core::clone::Clone::clone(&self.dump_mir_exclude_pass_number),
            dump_mir_graphviz: ::core::clone::Clone::clone(&self.dump_mir_graphviz),
            dump_mono_stats: ::core::clone::Clone::clone(&self.dump_mono_stats),
            dump_mono_stats_format: ::core::clone::Clone::clone(&self.dump_mono_stats_format),
            dwarf_version: ::core::clone::Clone::clone(&self.dwarf_version),
            dylib_lto: ::core::clone::Clone::clone(&self.dylib_lto),
            eagerly_emit_delayed_bugs: ::core::clone::Clone::clone(&self.eagerly_emit_delayed_bugs),
            ehcont_guard: ::core::clone::Clone::clone(&self.ehcont_guard),
            embed_metadata: ::core::clone::Clone::clone(&self.embed_metadata),
            embed_source: ::core::clone::Clone::clone(&self.embed_source),
            emit_stack_sizes: ::core::clone::Clone::clone(&self.emit_stack_sizes),
            enforce_type_length_limit: ::core::clone::Clone::clone(&self.enforce_type_length_limit),
            experimental_default_bounds: ::core::clone::Clone::clone(&self.experimental_default_bounds),
            export_executable_symbols: ::core::clone::Clone::clone(&self.export_executable_symbols),
            external_clangrt: ::core::clone::Clone::clone(&self.external_clangrt),
            extra_const_ub_checks: ::core::clone::Clone::clone(&self.extra_const_ub_checks),
            fewer_names: ::core::clone::Clone::clone(&self.fewer_names),
            fixed_x18: ::core::clone::Clone::clone(&self.fixed_x18),
            flatten_format_args: ::core::clone::Clone::clone(&self.flatten_format_args),
            fmt_debug: ::core::clone::Clone::clone(&self.fmt_debug),
            force_unstable_if_unmarked: ::core::clone::Clone::clone(&self.force_unstable_if_unmarked),
            function_return: ::core::clone::Clone::clone(&self.function_return),
            function_sections: ::core::clone::Clone::clone(&self.function_sections),
            future_incompat_test: ::core::clone::Clone::clone(&self.future_incompat_test),
            graphviz_dark_mode: ::core::clone::Clone::clone(&self.graphviz_dark_mode),
            graphviz_font: ::core::clone::Clone::clone(&self.graphviz_font),
            has_thread_local: ::core::clone::Clone::clone(&self.has_thread_local),
            help: ::core::clone::Clone::clone(&self.help),
            higher_ranked_assumptions: ::core::clone::Clone::clone(&self.higher_ranked_assumptions),
            hint_mostly_unused: ::core::clone::Clone::clone(&self.hint_mostly_unused),
            human_readable_cgu_names: ::core::clone::Clone::clone(&self.human_readable_cgu_names),
            identify_regions: ::core::clone::Clone::clone(&self.identify_regions),
            ignore_directory_in_diagnostics_source_blocks: ::core::clone::Clone::clone(&self.ignore_directory_in_diagnostics_source_blocks),
            incremental_ignore_spans: ::core::clone::Clone::clone(&self.incremental_ignore_spans),
            incremental_info: ::core::clone::Clone::clone(&self.incremental_info),
            incremental_verify_ich: ::core::clone::Clone::clone(&self.incremental_verify_ich),
            indirect_branch_cs_prefix: ::core::clone::Clone::clone(&self.indirect_branch_cs_prefix),
            inline_llvm: ::core::clone::Clone::clone(&self.inline_llvm),
            inline_mir: ::core::clone::Clone::clone(&self.inline_mir),
            inline_mir_forwarder_threshold: ::core::clone::Clone::clone(&self.inline_mir_forwarder_threshold),
            inline_mir_hint_threshold: ::core::clone::Clone::clone(&self.inline_mir_hint_threshold),
            inline_mir_preserve_debug: ::core::clone::Clone::clone(&self.inline_mir_preserve_debug),
            inline_mir_threshold: ::core::clone::Clone::clone(&self.inline_mir_threshold),
            input_stats: ::core::clone::Clone::clone(&self.input_stats),
            instrument_mcount: ::core::clone::Clone::clone(&self.instrument_mcount),
            instrument_xray: ::core::clone::Clone::clone(&self.instrument_xray),
            internal_testing_features: ::core::clone::Clone::clone(&self.internal_testing_features),
            large_data_threshold: ::core::clone::Clone::clone(&self.large_data_threshold),
            layout_seed: ::core::clone::Clone::clone(&self.layout_seed),
            link_directives: ::core::clone::Clone::clone(&self.link_directives),
            link_native_libraries: ::core::clone::Clone::clone(&self.link_native_libraries),
            link_only: ::core::clone::Clone::clone(&self.link_only),
            lint_llvm_ir: ::core::clone::Clone::clone(&self.lint_llvm_ir),
            lint_mir: ::core::clone::Clone::clone(&self.lint_mir),
            lint_rust_version: ::core::clone::Clone::clone(&self.lint_rust_version),
            llvm_module_flag: ::core::clone::Clone::clone(&self.llvm_module_flag),
            llvm_plugins: ::core::clone::Clone::clone(&self.llvm_plugins),
            llvm_time_trace: ::core::clone::Clone::clone(&self.llvm_time_trace),
            llvm_writable: ::core::clone::Clone::clone(&self.llvm_writable),
            location_detail: ::core::clone::Clone::clone(&self.location_detail),
            ls: ::core::clone::Clone::clone(&self.ls),
            macro_backtrace: ::core::clone::Clone::clone(&self.macro_backtrace),
            macro_stats: ::core::clone::Clone::clone(&self.macro_stats),
            maximal_hir_to_mir_coverage: ::core::clone::Clone::clone(&self.maximal_hir_to_mir_coverage),
            merge_functions: ::core::clone::Clone::clone(&self.merge_functions),
            meta_stats: ::core::clone::Clone::clone(&self.meta_stats),
            metrics_dir: ::core::clone::Clone::clone(&self.metrics_dir),
            min_function_alignment: ::core::clone::Clone::clone(&self.min_function_alignment),
            min_recursion_limit: ::core::clone::Clone::clone(&self.min_recursion_limit),
            mir_enable_passes: ::core::clone::Clone::clone(&self.mir_enable_passes),
            mir_include_spans: ::core::clone::Clone::clone(&self.mir_include_spans),
            mir_opt_bisect_limit: ::core::clone::Clone::clone(&self.mir_opt_bisect_limit),
            mir_opt_level: ::core::clone::Clone::clone(&self.mir_opt_level),
            mir_preserve_ub: ::core::clone::Clone::clone(&self.mir_preserve_ub),
            mir_strip_debuginfo: ::core::clone::Clone::clone(&self.mir_strip_debuginfo),
            move_size_limit: ::core::clone::Clone::clone(&self.move_size_limit),
            mutable_noalias: ::core::clone::Clone::clone(&self.mutable_noalias),
            namespaced_crates: ::core::clone::Clone::clone(&self.namespaced_crates),
            next_solver: ::core::clone::Clone::clone(&self.next_solver),
            nll_facts: ::core::clone::Clone::clone(&self.nll_facts),
            nll_facts_dir: ::core::clone::Clone::clone(&self.nll_facts_dir),
            no_analysis: ::core::clone::Clone::clone(&self.no_analysis),
            no_codegen: ::core::clone::Clone::clone(&self.no_codegen),
            no_generate_arange_section: ::core::clone::Clone::clone(&self.no_generate_arange_section),
            no_implied_bounds_compat: ::core::clone::Clone::clone(&self.no_implied_bounds_compat),
            no_leak_check: ::core::clone::Clone::clone(&self.no_leak_check),
            no_link: ::core::clone::Clone::clone(&self.no_link),
            no_parallel_backend: ::core::clone::Clone::clone(&self.no_parallel_backend),
            no_profiler_runtime: ::core::clone::Clone::clone(&self.no_profiler_runtime),
            no_steal_thir: ::core::clone::Clone::clone(&self.no_steal_thir),
            no_trait_vptr: ::core::clone::Clone::clone(&self.no_trait_vptr),
            no_unique_section_names: ::core::clone::Clone::clone(&self.no_unique_section_names),
            normalize_docs: ::core::clone::Clone::clone(&self.normalize_docs),
            offload: ::core::clone::Clone::clone(&self.offload),
            on_broken_pipe: ::core::clone::Clone::clone(&self.on_broken_pipe),
            osx_rpath_install_name: ::core::clone::Clone::clone(&self.osx_rpath_install_name),
            packed_bundled_libs: ::core::clone::Clone::clone(&self.packed_bundled_libs),
            packed_stack: ::core::clone::Clone::clone(&self.packed_stack),
            panic_abort_tests: ::core::clone::Clone::clone(&self.panic_abort_tests),
            panic_in_drop: ::core::clone::Clone::clone(&self.panic_in_drop),
            parse_crate_root_only: ::core::clone::Clone::clone(&self.parse_crate_root_only),
            patchable_function_entry: ::core::clone::Clone::clone(&self.patchable_function_entry),
            plt: ::core::clone::Clone::clone(&self.plt),
            polonius: ::core::clone::Clone::clone(&self.polonius),
            pre_link_arg: ::core::clone::Clone::clone(&self.pre_link_arg),
            pre_link_args: ::core::clone::Clone::clone(&self.pre_link_args),
            precise_enum_drop_elaboration: ::core::clone::Clone::clone(&self.precise_enum_drop_elaboration),
            print_codegen_stats: ::core::clone::Clone::clone(&self.print_codegen_stats),
            print_codegen_stats_json: ::core::clone::Clone::clone(&self.print_codegen_stats_json),
            print_llvm_passes: ::core::clone::Clone::clone(&self.print_llvm_passes),
            print_mono_items: ::core::clone::Clone::clone(&self.print_mono_items),
            print_type_sizes: ::core::clone::Clone::clone(&self.print_type_sizes),
            proc_macro_backtrace: ::core::clone::Clone::clone(&self.proc_macro_backtrace),
            proc_macro_execution_strategy: ::core::clone::Clone::clone(&self.proc_macro_execution_strategy),
            profile_closures: ::core::clone::Clone::clone(&self.profile_closures),
            profile_sample_use: ::core::clone::Clone::clone(&self.profile_sample_use),
            profiler_runtime: ::core::clone::Clone::clone(&self.profiler_runtime),
            query_dep_graph: ::core::clone::Clone::clone(&self.query_dep_graph),
            randomize_layout: ::core::clone::Clone::clone(&self.randomize_layout),
            reg_struct_return: ::core::clone::Clone::clone(&self.reg_struct_return),
            regparm: ::core::clone::Clone::clone(&self.regparm),
            relax_elf_relocations: ::core::clone::Clone::clone(&self.relax_elf_relocations),
            remap_cwd_prefix: ::core::clone::Clone::clone(&self.remap_cwd_prefix),
            remark_dir: ::core::clone::Clone::clone(&self.remark_dir),
            retpoline: ::core::clone::Clone::clone(&self.retpoline),
            retpoline_external_thunk: ::core::clone::Clone::clone(&self.retpoline_external_thunk),
            sanitizer: ::core::clone::Clone::clone(&self.sanitizer),
            sanitizer_cfi_canonical_jump_tables: ::core::clone::Clone::clone(&self.sanitizer_cfi_canonical_jump_tables),
            sanitizer_cfi_generalize_pointers: ::core::clone::Clone::clone(&self.sanitizer_cfi_generalize_pointers),
            sanitizer_cfi_normalize_integers: ::core::clone::Clone::clone(&self.sanitizer_cfi_normalize_integers),
            sanitizer_dataflow_abilist: ::core::clone::Clone::clone(&self.sanitizer_dataflow_abilist),
            sanitizer_kcfi_arity: ::core::clone::Clone::clone(&self.sanitizer_kcfi_arity),
            sanitizer_memory_track_origins: ::core::clone::Clone::clone(&self.sanitizer_memory_track_origins),
            sanitizer_recover: ::core::clone::Clone::clone(&self.sanitizer_recover),
            saturating_float_casts: ::core::clone::Clone::clone(&self.saturating_float_casts),
            self_profile: ::core::clone::Clone::clone(&self.self_profile),
            self_profile_counter: ::core::clone::Clone::clone(&self.self_profile_counter),
            self_profile_events: ::core::clone::Clone::clone(&self.self_profile_events),
            share_generics: ::core::clone::Clone::clone(&self.share_generics),
            shell_argfiles: ::core::clone::Clone::clone(&self.shell_argfiles),
            simulate_remapped_rust_src_base: ::core::clone::Clone::clone(&self.simulate_remapped_rust_src_base),
            small_data_threshold: ::core::clone::Clone::clone(&self.small_data_threshold),
            span_debug: ::core::clone::Clone::clone(&self.span_debug),
            span_free_formats: ::core::clone::Clone::clone(&self.span_free_formats),
            split_dwarf_inlining: ::core::clone::Clone::clone(&self.split_dwarf_inlining),
            split_dwarf_kind: ::core::clone::Clone::clone(&self.split_dwarf_kind),
            split_dwarf_out_dir: ::core::clone::Clone::clone(&self.split_dwarf_out_dir),
            split_lto_unit: ::core::clone::Clone::clone(&self.split_lto_unit),
            src_hash_algorithm: ::core::clone::Clone::clone(&self.src_hash_algorithm),
            stack_protector: ::core::clone::Clone::clone(&self.stack_protector),
            staticlib_allow_rdylib_deps: ::core::clone::Clone::clone(&self.staticlib_allow_rdylib_deps),
            staticlib_hide_internal_symbols: ::core::clone::Clone::clone(&self.staticlib_hide_internal_symbols),
            staticlib_rename_internal_symbols: ::core::clone::Clone::clone(&self.staticlib_rename_internal_symbols),
            staticlib_prefer_dynamic: ::core::clone::Clone::clone(&self.staticlib_prefer_dynamic),
            strict_init_checks: ::core::clone::Clone::clone(&self.strict_init_checks),
            teach: ::core::clone::Clone::clone(&self.teach),
            temps_dir: ::core::clone::Clone::clone(&self.temps_dir),
            terminal_urls: ::core::clone::Clone::clone(&self.terminal_urls),
            thinlto: ::core::clone::Clone::clone(&self.thinlto),
            threads: ::core::clone::Clone::clone(&self.threads),
            time_llvm_passes: ::core::clone::Clone::clone(&self.time_llvm_passes),
            time_passes: ::core::clone::Clone::clone(&self.time_passes),
            time_passes_format: ::core::clone::Clone::clone(&self.time_passes_format),
            tiny_const_eval_limit: ::core::clone::Clone::clone(&self.tiny_const_eval_limit),
            tls_model: ::core::clone::Clone::clone(&self.tls_model),
            trace_macros: ::core::clone::Clone::clone(&self.trace_macros),
            track_diagnostics: ::core::clone::Clone::clone(&self.track_diagnostics),
            translate_remapped_path_to_local_path: ::core::clone::Clone::clone(&self.translate_remapped_path_to_local_path),
            trap_unreachable: ::core::clone::Clone::clone(&self.trap_unreachable),
            treat_err_as_bug: ::core::clone::Clone::clone(&self.treat_err_as_bug),
            trim_diagnostic_paths: ::core::clone::Clone::clone(&self.trim_diagnostic_paths),
            tune_cpu: ::core::clone::Clone::clone(&self.tune_cpu),
            typing_mode_post_typeck_until_borrowck: ::core::clone::Clone::clone(&self.typing_mode_post_typeck_until_borrowck),
            ub_checks: ::core::clone::Clone::clone(&self.ub_checks),
            ui_testing: ::core::clone::Clone::clone(&self.ui_testing),
            uninit_const_chunk_threshold: ::core::clone::Clone::clone(&self.uninit_const_chunk_threshold),
            unleash_the_miri_inside_of_you: ::core::clone::Clone::clone(&self.unleash_the_miri_inside_of_you),
            unpretty: ::core::clone::Clone::clone(&self.unpretty),
            unsound_mir_opts: ::core::clone::Clone::clone(&self.unsound_mir_opts),
            unstable_options: ::core::clone::Clone::clone(&self.unstable_options),
            use_ctors_section: ::core::clone::Clone::clone(&self.use_ctors_section),
            use_sync_unwind: ::core::clone::Clone::clone(&self.use_sync_unwind),
            validate_mir: ::core::clone::Clone::clone(&self.validate_mir),
            verbose_asm: ::core::clone::Clone::clone(&self.verbose_asm),
            verbose_internals: ::core::clone::Clone::clone(&self.verbose_internals),
            verify_llvm_ir: ::core::clone::Clone::clone(&self.verify_llvm_ir),
            virtual_function_elimination: ::core::clone::Clone::clone(&self.virtual_function_elimination),
            wasi_exec_model: ::core::clone::Clone::clone(&self.wasi_exec_model),
            wasm_c_abi: ::core::clone::Clone::clone(&self.wasm_c_abi),
            write_long_types_to_disk: ::core::clone::Clone::clone(&self.write_long_types_to_disk),
        }
    }
}
pub enum UnstableOptionsTargetModifiers {
    BranchProtection,
    FixedX18,
    IndirectBranchCsPrefix,
    RegStructReturn,
    Regparm,
    Retpoline,
    RetpolineExternalThunk,
    Sanitizer,
    SanitizerCfiNormalizeIntegers,
}
#[automatically_derived]
impl ::core::marker::StructuralPartialEq for UnstableOptionsTargetModifiers {
}
#[automatically_derived]
impl ::core::cmp::PartialEq for UnstableOptionsTargetModifiers {
    #[inline]
    fn eq(&self, other: &UnstableOptionsTargetModifiers) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}
#[automatically_derived]
impl ::core::cmp::Eq for UnstableOptionsTargetModifiers {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}
#[automatically_derived]
impl ::core::cmp::PartialOrd for UnstableOptionsTargetModifiers {
    #[inline]
    fn partial_cmp(&self, other: &UnstableOptionsTargetModifiers)
        -> ::core::option::Option<::core::cmp::Ordering> {
        ::core::option::Option::Some(::core::cmp::Ord::cmp(self, other))
    }
}
#[automatically_derived]
impl ::core::cmp::Ord for UnstableOptionsTargetModifiers {
    #[inline]
    fn cmp(&self, other: &UnstableOptionsTargetModifiers)
        -> ::core::cmp::Ordering {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        ::core::cmp::Ord::cmp(&__self_discr, &__arg1_discr)
    }
}
#[automatically_derived]
impl ::core::fmt::Debug for UnstableOptionsTargetModifiers {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                UnstableOptionsTargetModifiers::BranchProtection =>
                    "BranchProtection",
                UnstableOptionsTargetModifiers::FixedX18 => "FixedX18",
                UnstableOptionsTargetModifiers::IndirectBranchCsPrefix =>
                    "IndirectBranchCsPrefix",
                UnstableOptionsTargetModifiers::RegStructReturn =>
                    "RegStructReturn",
                UnstableOptionsTargetModifiers::Regparm => "Regparm",
                UnstableOptionsTargetModifiers::Retpoline => "Retpoline",
                UnstableOptionsTargetModifiers::RetpolineExternalThunk =>
                    "RetpolineExternalThunk",
                UnstableOptionsTargetModifiers::Sanitizer => "Sanitizer",
                UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers
                    => "SanitizerCfiNormalizeIntegers",
            })
    }
}
#[automatically_derived]
impl ::core::marker::Copy for UnstableOptionsTargetModifiers { }
#[automatically_derived]
#[doc(hidden)]
unsafe impl ::core::clone::TrivialClone for UnstableOptionsTargetModifiers { }
#[automatically_derived]
impl ::core::clone::Clone for UnstableOptionsTargetModifiers {
    #[inline]
    fn clone(&self) -> UnstableOptionsTargetModifiers { *self }
}
const _: () =
    {
        impl<__E: ::rustc_span::SpanEncoder> ::rustc_serialize::Encodable<__E>
            for UnstableOptionsTargetModifiers {
            fn encode(&self, __encoder: &mut __E) {
                let disc =
                    match *self {
                        UnstableOptionsTargetModifiers::BranchProtection => {
                            0usize
                        }
                        UnstableOptionsTargetModifiers::FixedX18 => { 1usize }
                        UnstableOptionsTargetModifiers::IndirectBranchCsPrefix => {
                            2usize
                        }
                        UnstableOptionsTargetModifiers::RegStructReturn => {
                            3usize
                        }
                        UnstableOptionsTargetModifiers::Regparm => { 4usize }
                        UnstableOptionsTargetModifiers::Retpoline => { 5usize }
                        UnstableOptionsTargetModifiers::RetpolineExternalThunk => {
                            6usize
                        }
                        UnstableOptionsTargetModifiers::Sanitizer => { 7usize }
                        UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers
                            => {
                            8usize
                        }
                    };
                ::rustc_serialize::Encoder::emit_u8(__encoder, disc as u8);
                match *self {
                    UnstableOptionsTargetModifiers::BranchProtection => {}
                    UnstableOptionsTargetModifiers::FixedX18 => {}
                    UnstableOptionsTargetModifiers::IndirectBranchCsPrefix => {}
                    UnstableOptionsTargetModifiers::RegStructReturn => {}
                    UnstableOptionsTargetModifiers::Regparm => {}
                    UnstableOptionsTargetModifiers::Retpoline => {}
                    UnstableOptionsTargetModifiers::RetpolineExternalThunk => {}
                    UnstableOptionsTargetModifiers::Sanitizer => {}
                    UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers
                        => {}
                }
            }
        }
    };
const _: () =
    {
        impl<__D: ::rustc_span::BlobDecoder> ::rustc_serialize::Decodable<__D>
            for UnstableOptionsTargetModifiers {
            fn decode(__decoder: &mut __D) -> Self {
                match ::rustc_serialize::Decoder::read_u8(__decoder) as usize
                    {
                    0usize => {
                        UnstableOptionsTargetModifiers::BranchProtection
                    }
                    1usize => { UnstableOptionsTargetModifiers::FixedX18 }
                    2usize => {
                        UnstableOptionsTargetModifiers::IndirectBranchCsPrefix
                    }
                    3usize => {
                        UnstableOptionsTargetModifiers::RegStructReturn
                    }
                    4usize => { UnstableOptionsTargetModifiers::Regparm }
                    5usize => { UnstableOptionsTargetModifiers::Retpoline }
                    6usize => {
                        UnstableOptionsTargetModifiers::RetpolineExternalThunk
                    }
                    7usize => { UnstableOptionsTargetModifiers::Sanitizer }
                    8usize => {
                        UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers
                    }
                    n => {
                        ::core::panicking::panic_fmt(format_args!("invalid enum variant tag while decoding `UnstableOptionsTargetModifiers`, expected 0..9, actual {0}",
                                n));
                    }
                }
            }
        }
    };
impl UnstableOptionsTargetModifiers {
    pub fn reparse(&self, _user_value: &str) -> ExtendedTargetModifierInfo {
        match self {
            Self::BranchProtection => {
                let mut parsed: Option<BranchProtection> = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_branch_protection(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "branch_protection".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::FixedX18 => {
                let mut parsed: bool = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "fixed_x18".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::IndirectBranchCsPrefix => {
                let mut parsed: bool = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "indirect_branch_cs_prefix".to_string().replace('_',
                        "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::RegStructReturn => {
                let mut parsed: bool = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "reg_struct_return".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::Regparm => {
                let mut parsed: Option<u32> = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_opt_number(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "regparm".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::Retpoline => {
                let mut parsed: bool = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "retpoline".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::RetpolineExternalThunk => {
                let mut parsed: bool = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "retpoline_external_thunk".to_string().replace('_',
                        "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::Sanitizer => {
                let mut parsed: SanitizerSet = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_sanitizers(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "sanitizer".to_string().replace('_', "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
            Self::SanitizerCfiNormalizeIntegers => {
                let mut parsed: Option<bool> = Default::default();
                let val =
                    if _user_value.is_empty() {
                        None
                    } else { Some(_user_value) };
                parse::parse_opt_bool(&mut parsed, val);
                ExtendedTargetModifierInfo {
                    prefix: "Z".to_string(),
                    name: "sanitizer_cfi_normalize_integers".to_string().replace('_',
                        "-"),
                    tech_value: ::alloc::__export::must_use({
                            ::alloc::fmt::format(format_args!("{0:?}", parsed))
                        }),
                }
            }
                #[allow(unreachable_patterns)]
                _ => {
                ::core::panicking::panic_fmt(format_args!("unknown target modifier option: {0:?}",
                        *self));
            }
        }
    }
    pub fn is_target_modifier(flag_name: &str) -> bool {
        match flag_name.replace('-', "_").as_str() {
            "branch_protection" => true,
            "fixed_x18" => true,
            "indirect_branch_cs_prefix" => true,
            "reg_struct_return" => true,
            "regparm" => true,
            "retpoline" => true,
            "retpoline_external_thunk" => true,
            "sanitizer" => true,
            "sanitizer_cfi_normalize_integers" => true,
            _ => false,
        }
    }
}
impl Default for UnstableOptions {
    fn default() -> UnstableOptions {
        UnstableOptions {
            allow_features: None,
            allow_partial_mitigations: (),
            always_encode_mir: false,
            annotate_moves: AnnotateMoves::Disabled,
            assert_incr_state: None,
            assume_incomplete_release: false,
            assumptions_on_binders: false,
            autodiff: Vec::new(),
            binary_dep_depinfo: false,
            box_noalias: true,
            branch_protection: None,
            build_sdylib_interface: false,
            cache_proc_macros: false,
            cf_protection: CFProtection::None,
            check_cfg_all_expected: false,
            checksum_hash_algorithm: None,
            codegen_backend: None,
            codegen_emit_retag: None,
            codegen_source_order: false,
            contract_checks: None,
            coverage_options: CoverageOptions::default(),
            crate_attr: Vec::new(),
            cross_crate_inline_threshold: InliningThreshold::Sometimes(100),
            debug_info_type_line_numbers: false,
            debuginfo_compression: DebugInfoCompression::None,
            debuginfo_for_profiling: false,
            deduplicate_diagnostics: true,
            default_visibility: None,
            deny_partial_mitigations: (),
            dep_info_omit_d_target: false,
            direct_access_external_data: None,
            disable_fast_paths: false,
            disable_incr_comp_backend_caching: false,
            dual_proc_macros: false,
            dump_dep_graph: false,
            dump_mir: None,
            dump_mir_dataflow: false,
            dump_mir_dir: "mir_dump".to_string(),
            dump_mir_exclude_alloc_bytes: false,
            dump_mir_exclude_pass_number: false,
            dump_mir_graphviz: false,
            dump_mono_stats: SwitchWithOptPath::Disabled,
            dump_mono_stats_format: DumpMonoStatsFormat::Markdown,
            dwarf_version: None,
            dylib_lto: false,
            eagerly_emit_delayed_bugs: false,
            ehcont_guard: false,
            embed_metadata: true,
            embed_source: false,
            emit_stack_sizes: false,
            enforce_type_length_limit: false,
            experimental_default_bounds: false,
            export_executable_symbols: false,
            external_clangrt: false,
            extra_const_ub_checks: false,
            fewer_names: None,
            fixed_x18: false,
            flatten_format_args: true,
            fmt_debug: FmtDebug::Full,
            force_unstable_if_unmarked: false,
            function_return: FunctionReturn::default(),
            function_sections: None,
            future_incompat_test: false,
            graphviz_dark_mode: false,
            graphviz_font: "Courier, monospace".to_string(),
            has_thread_local: None,
            help: false,
            higher_ranked_assumptions: false,
            hint_mostly_unused: false,
            human_readable_cgu_names: false,
            identify_regions: false,
            ignore_directory_in_diagnostics_source_blocks: Vec::new(),
            incremental_ignore_spans: false,
            incremental_info: false,
            incremental_verify_ich: false,
            indirect_branch_cs_prefix: false,
            inline_llvm: true,
            inline_mir: None,
            inline_mir_forwarder_threshold: None,
            inline_mir_hint_threshold: None,
            inline_mir_preserve_debug: None,
            inline_mir_threshold: None,
            input_stats: false,
            instrument_mcount: false,
            instrument_xray: None,
            internal_testing_features: false,
            large_data_threshold: None,
            layout_seed: None,
            link_directives: true,
            link_native_libraries: true,
            link_only: false,
            lint_llvm_ir: false,
            lint_mir: false,
            lint_rust_version: None,
            llvm_module_flag: Vec::new(),
            llvm_plugins: Vec::new(),
            llvm_time_trace: false,
            llvm_writable: false,
            location_detail: LocationDetail::all(),
            ls: Vec::new(),
            macro_backtrace: false,
            macro_stats: false,
            maximal_hir_to_mir_coverage: false,
            merge_functions: None,
            meta_stats: false,
            metrics_dir: None,
            min_function_alignment: None,
            min_recursion_limit: None,
            mir_enable_passes: Vec::new(),
            mir_include_spans: MirIncludeSpans::default(),
            mir_opt_bisect_limit: None,
            mir_opt_level: None,
            mir_preserve_ub: false,
            mir_strip_debuginfo: MirStripDebugInfo::None,
            move_size_limit: None,
            mutable_noalias: true,
            namespaced_crates: false,
            next_solver: NextSolverConfig::default(),
            nll_facts: false,
            nll_facts_dir: "nll-facts".to_string(),
            no_analysis: false,
            no_codegen: false,
            no_generate_arange_section: false,
            no_implied_bounds_compat: false,
            no_leak_check: false,
            no_link: false,
            no_parallel_backend: false,
            no_profiler_runtime: false,
            no_steal_thir: false,
            no_trait_vptr: false,
            no_unique_section_names: false,
            normalize_docs: false,
            offload: Vec::new(),
            on_broken_pipe: OnBrokenPipe::Default,
            osx_rpath_install_name: false,
            packed_bundled_libs: false,
            packed_stack: false,
            panic_abort_tests: false,
            panic_in_drop: PanicStrategy::Unwind,
            parse_crate_root_only: false,
            patchable_function_entry: PatchableFunctionEntry::default(),
            plt: None,
            polonius: Polonius::default(),
            pre_link_arg: (),
            pre_link_args: Vec::new(),
            precise_enum_drop_elaboration: true,
            print_codegen_stats: false,
            print_codegen_stats_json: None,
            print_llvm_passes: false,
            print_mono_items: false,
            print_type_sizes: false,
            proc_macro_backtrace: false,
            proc_macro_execution_strategy: ProcMacroExecutionStrategy::SameThread,
            profile_closures: false,
            profile_sample_use: None,
            profiler_runtime: String::from("profiler_builtins"),
            query_dep_graph: false,
            randomize_layout: false,
            reg_struct_return: false,
            regparm: None,
            relax_elf_relocations: None,
            remap_cwd_prefix: None,
            remark_dir: None,
            retpoline: false,
            retpoline_external_thunk: false,
            sanitizer: SanitizerSet::empty(),
            sanitizer_cfi_canonical_jump_tables: Some(true),
            sanitizer_cfi_generalize_pointers: None,
            sanitizer_cfi_normalize_integers: None,
            sanitizer_dataflow_abilist: Vec::new(),
            sanitizer_kcfi_arity: None,
            sanitizer_memory_track_origins: 0,
            sanitizer_recover: SanitizerSet::empty(),
            saturating_float_casts: None,
            self_profile: SwitchWithOptPath::Disabled,
            self_profile_counter: "wall-time".to_string(),
            self_profile_events: None,
            share_generics: None,
            shell_argfiles: false,
            simulate_remapped_rust_src_base: None,
            small_data_threshold: None,
            span_debug: false,
            span_free_formats: false,
            split_dwarf_inlining: false,
            split_dwarf_kind: SplitDwarfKind::Split,
            split_dwarf_out_dir: None,
            split_lto_unit: None,
            src_hash_algorithm: None,
            stack_protector: StackProtector::None,
            staticlib_allow_rdylib_deps: false,
            staticlib_hide_internal_symbols: false,
            staticlib_rename_internal_symbols: false,
            staticlib_prefer_dynamic: false,
            strict_init_checks: false,
            teach: false,
            temps_dir: None,
            terminal_urls: TerminalUrl::No,
            thinlto: None,
            threads: None,
            time_llvm_passes: false,
            time_passes: false,
            time_passes_format: TimePassesFormat::Text,
            tiny_const_eval_limit: false,
            tls_model: None,
            trace_macros: false,
            track_diagnostics: false,
            translate_remapped_path_to_local_path: true,
            trap_unreachable: None,
            treat_err_as_bug: None,
            trim_diagnostic_paths: true,
            tune_cpu: None,
            typing_mode_post_typeck_until_borrowck: false,
            ub_checks: None,
            ui_testing: false,
            uninit_const_chunk_threshold: 16,
            unleash_the_miri_inside_of_you: false,
            unpretty: None,
            unsound_mir_opts: false,
            unstable_options: false,
            use_ctors_section: None,
            use_sync_unwind: None,
            validate_mir: false,
            verbose_asm: false,
            verbose_internals: false,
            verify_llvm_ir: false,
            virtual_function_elimination: false,
            wasi_exec_model: None,
            wasm_c_abi: (),
            write_long_types_to_disk: true,
        }
    }
}
impl UnstableOptions {
    pub fn build(early_dcx: &EarlyDiagCtxt, matches: &getopts::Matches,
        target_modifiers: &mut CollectedOptions) -> UnstableOptions {
        build_options(early_dcx, matches, target_modifiers, Z_OPTIONS, "Z",
            "unstable")
    }
    fn dep_tracking_hash(&self, for_crate_hash: bool,
        error_format: ErrorOutputType) -> Hash64 {
        let mut sub_hashes = BTreeMap::new();
        {
            if (&mut sub_hashes).insert("allow_features",
                        &self.allow_features as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "allow_features"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("always_encode_mir",
                        &self.always_encode_mir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "always_encode_mir"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("annotate_moves",
                        &self.annotate_moves as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "annotate_moves"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("assume_incomplete_release",
                        &self.assume_incomplete_release as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "assume_incomplete_release"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("assumptions_on_binders",
                        &self.assumptions_on_binders as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "assumptions_on_binders"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("autodiff",
                        &self.autodiff as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "autodiff"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("binary_dep_depinfo",
                        &self.binary_dep_depinfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "binary_dep_depinfo"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("box_noalias",
                        &self.box_noalias as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "box_noalias"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("branch_protection",
                        &self.branch_protection as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "branch_protection"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("cache_proc_macros",
                        &self.cache_proc_macros as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "cache_proc_macros"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("cf_protection",
                        &self.cf_protection as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "cf_protection"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("checksum_hash_algorithm",
                        &self.checksum_hash_algorithm as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "checksum_hash_algorithm"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("codegen_backend",
                        &self.codegen_backend as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "codegen_backend"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("codegen_emit_retag",
                        &self.codegen_emit_retag as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "codegen_emit_retag"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("contract_checks",
                        &self.contract_checks as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "contract_checks"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("coverage_options",
                        &self.coverage_options as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "coverage_options"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("crate_attr",
                        &self.crate_attr as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "crate_attr"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("cross_crate_inline_threshold",
                        &self.cross_crate_inline_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "cross_crate_inline_threshold"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debug_info_type_line_numbers",
                        &self.debug_info_type_line_numbers as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debug_info_type_line_numbers"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debuginfo_compression",
                        &self.debuginfo_compression as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debuginfo_compression"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("debuginfo_for_profiling",
                        &self.debuginfo_for_profiling as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "debuginfo_for_profiling"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("default_visibility",
                        &self.default_visibility as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "default_visibility"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("dep_info_omit_d_target",
                        &self.dep_info_omit_d_target as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "dep_info_omit_d_target"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("direct_access_external_data",
                        &self.direct_access_external_data as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "direct_access_external_data"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("disable_fast_paths",
                        &self.disable_fast_paths as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "disable_fast_paths"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("disable_incr_comp_backend_caching",
                        &self.disable_incr_comp_backend_caching as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "disable_incr_comp_backend_caching"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("dual_proc_macros",
                        &self.dual_proc_macros as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "dual_proc_macros"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("dwarf_version",
                        &self.dwarf_version as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "dwarf_version"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("ehcont_guard",
                        &self.ehcont_guard as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "ehcont_guard"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("embed_metadata",
                        &self.embed_metadata as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "embed_metadata"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("embed_source",
                        &self.embed_source as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "embed_source"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("enforce_type_length_limit",
                        &self.enforce_type_length_limit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "enforce_type_length_limit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("experimental_default_bounds",
                        &self.experimental_default_bounds as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "experimental_default_bounds"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("export_executable_symbols",
                        &self.export_executable_symbols as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "export_executable_symbols"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("extra_const_ub_checks",
                        &self.extra_const_ub_checks as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "extra_const_ub_checks"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("fewer_names",
                        &self.fewer_names as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "fewer_names"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("fixed_x18",
                        &self.fixed_x18 as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "fixed_x18"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("flatten_format_args",
                        &self.flatten_format_args as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "flatten_format_args"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("fmt_debug",
                        &self.fmt_debug as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "fmt_debug"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("force_unstable_if_unmarked",
                        &self.force_unstable_if_unmarked as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "force_unstable_if_unmarked"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("function_return",
                        &self.function_return as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "function_return"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("function_sections",
                        &self.function_sections as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "function_sections"));
                }
            }
        };
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("has_thread_local",
                        &self.has_thread_local as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "has_thread_local"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("higher_ranked_assumptions",
                        &self.higher_ranked_assumptions as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "higher_ranked_assumptions"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("hint_mostly_unused",
                        &self.hint_mostly_unused as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "hint_mostly_unused"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("human_readable_cgu_names",
                        &self.human_readable_cgu_names as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "human_readable_cgu_names"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("incremental_ignore_spans",
                        &self.incremental_ignore_spans as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "incremental_ignore_spans"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("indirect_branch_cs_prefix",
                        &self.indirect_branch_cs_prefix as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "indirect_branch_cs_prefix"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_llvm",
                        &self.inline_llvm as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_llvm"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_mir",
                        &self.inline_mir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_mir"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_mir_forwarder_threshold",
                        &self.inline_mir_forwarder_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_mir_forwarder_threshold"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_mir_hint_threshold",
                        &self.inline_mir_hint_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_mir_hint_threshold"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_mir_preserve_debug",
                        &self.inline_mir_preserve_debug as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_mir_preserve_debug"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("inline_mir_threshold",
                        &self.inline_mir_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "inline_mir_threshold"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("instrument_mcount",
                        &self.instrument_mcount as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "instrument_mcount"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("instrument_xray",
                        &self.instrument_xray as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "instrument_xray"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("internal_testing_features",
                        &self.internal_testing_features as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "internal_testing_features"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("large_data_threshold",
                        &self.large_data_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "large_data_threshold"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("layout_seed",
                        &self.layout_seed as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "layout_seed"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("link_directives",
                        &self.link_directives as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "link_directives"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("link_only",
                        &self.link_only as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "link_only"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("lint_llvm_ir",
                        &self.lint_llvm_ir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "lint_llvm_ir"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("lint_rust_version",
                        &self.lint_rust_version as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "lint_rust_version"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("llvm_module_flag",
                        &self.llvm_module_flag as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "llvm_module_flag"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("llvm_plugins",
                        &self.llvm_plugins as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "llvm_plugins"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("llvm_writable",
                        &self.llvm_writable as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "llvm_writable"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("location_detail",
                        &self.location_detail as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "location_detail"));
                }
            }
        };
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("maximal_hir_to_mir_coverage",
                        &self.maximal_hir_to_mir_coverage as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "maximal_hir_to_mir_coverage"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("merge_functions",
                        &self.merge_functions as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "merge_functions"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("min_function_alignment",
                        &self.min_function_alignment as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "min_function_alignment"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("min_recursion_limit",
                        &self.min_recursion_limit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "min_recursion_limit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("mir_enable_passes",
                        &self.mir_enable_passes as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mir_enable_passes"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("mir_opt_bisect_limit",
                        &self.mir_opt_bisect_limit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mir_opt_bisect_limit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("mir_opt_level",
                        &self.mir_opt_level as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mir_opt_level"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("mir_preserve_ub",
                        &self.mir_preserve_ub as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mir_preserve_ub"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("mir_strip_debuginfo",
                        &self.mir_strip_debuginfo as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mir_strip_debuginfo"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("move_size_limit",
                        &self.move_size_limit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "move_size_limit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("mutable_noalias",
                        &self.mutable_noalias as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "mutable_noalias"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("namespaced_crates",
                        &self.namespaced_crates as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "namespaced_crates"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("next_solver",
                        &self.next_solver as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "next_solver"));
                }
            }
        };
        {};
        {};
        {};
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("no_codegen",
                            &self.no_codegen as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "no_codegen"));
                    }
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_generate_arange_section",
                        &self.no_generate_arange_section as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_generate_arange_section"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_implied_bounds_compat",
                        &self.no_implied_bounds_compat as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_implied_bounds_compat"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("no_link",
                        &self.no_link as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_link"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("no_profiler_runtime",
                        &self.no_profiler_runtime as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_profiler_runtime"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("no_trait_vptr",
                        &self.no_trait_vptr as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_trait_vptr"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("no_unique_section_names",
                        &self.no_unique_section_names as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "no_unique_section_names"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("normalize_docs",
                        &self.normalize_docs as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "normalize_docs"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("offload",
                        &self.offload as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "offload"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("on_broken_pipe",
                        &self.on_broken_pipe as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "on_broken_pipe"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("osx_rpath_install_name",
                        &self.osx_rpath_install_name as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "osx_rpath_install_name"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("packed_bundled_libs",
                        &self.packed_bundled_libs as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "packed_bundled_libs"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("packed_stack",
                        &self.packed_stack as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "packed_stack"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("panic_abort_tests",
                        &self.panic_abort_tests as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "panic_abort_tests"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("panic_in_drop",
                        &self.panic_in_drop as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "panic_in_drop"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("patchable_function_entry",
                        &self.patchable_function_entry as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "patchable_function_entry"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("plt",
                        &self.plt as &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "plt"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("polonius",
                        &self.polonius as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "polonius"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("precise_enum_drop_elaboration",
                        &self.precise_enum_drop_elaboration as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "precise_enum_drop_elaboration"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("profile_sample_use",
                        &self.profile_sample_use as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "profile_sample_use"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("profiler_runtime",
                        &self.profiler_runtime as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "profiler_runtime"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("randomize_layout",
                        &self.randomize_layout as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "randomize_layout"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("reg_struct_return",
                        &self.reg_struct_return as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "reg_struct_return"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("regparm",
                        &self.regparm as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "regparm"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("relax_elf_relocations",
                        &self.relax_elf_relocations as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "relax_elf_relocations"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("remap_cwd_prefix",
                        &self.remap_cwd_prefix as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "remap_cwd_prefix"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("retpoline",
                        &self.retpoline as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "retpoline"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("retpoline_external_thunk",
                        &self.retpoline_external_thunk as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "retpoline_external_thunk"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer",
                        &self.sanitizer as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_cfi_canonical_jump_tables",
                        &self.sanitizer_cfi_canonical_jump_tables as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_cfi_canonical_jump_tables"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_cfi_generalize_pointers",
                        &self.sanitizer_cfi_generalize_pointers as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_cfi_generalize_pointers"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_cfi_normalize_integers",
                        &self.sanitizer_cfi_normalize_integers as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_cfi_normalize_integers"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_dataflow_abilist",
                        &self.sanitizer_dataflow_abilist as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_dataflow_abilist"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_kcfi_arity",
                        &self.sanitizer_kcfi_arity as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_kcfi_arity"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_memory_track_origins",
                        &self.sanitizer_memory_track_origins as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_memory_track_origins"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("sanitizer_recover",
                        &self.sanitizer_recover as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "sanitizer_recover"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("saturating_float_casts",
                        &self.saturating_float_casts as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "saturating_float_casts"));
                }
            }
        };
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("share_generics",
                        &self.share_generics as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "share_generics"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("simulate_remapped_rust_src_base",
                        &self.simulate_remapped_rust_src_base as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "simulate_remapped_rust_src_base"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("small_data_threshold",
                        &self.small_data_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "small_data_threshold"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("split_dwarf_inlining",
                        &self.split_dwarf_inlining as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "split_dwarf_inlining"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("split_dwarf_kind",
                        &self.split_dwarf_kind as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "split_dwarf_kind"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("split_dwarf_out_dir",
                        &self.split_dwarf_out_dir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "split_dwarf_out_dir"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("split_lto_unit",
                        &self.split_lto_unit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "split_lto_unit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("src_hash_algorithm",
                        &self.src_hash_algorithm as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "src_hash_algorithm"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("stack_protector",
                        &self.stack_protector as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "stack_protector"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("staticlib_allow_rdylib_deps",
                        &self.staticlib_allow_rdylib_deps as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "staticlib_allow_rdylib_deps"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("staticlib_hide_internal_symbols",
                        &self.staticlib_hide_internal_symbols as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "staticlib_hide_internal_symbols"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("staticlib_rename_internal_symbols",
                        &self.staticlib_rename_internal_symbols as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "staticlib_rename_internal_symbols"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("staticlib_prefer_dynamic",
                        &self.staticlib_prefer_dynamic as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "staticlib_prefer_dynamic"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("strict_init_checks",
                        &self.strict_init_checks as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "strict_init_checks"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("teach",
                        &self.teach as &dyn dep_tracking::DepTrackingHash).is_some()
                {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "teach"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("thinlto",
                        &self.thinlto as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "thinlto"));
                }
            }
        };
        {};
        {};
        {};
        {};
        {
            if (&mut sub_hashes).insert("tiny_const_eval_limit",
                        &self.tiny_const_eval_limit as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "tiny_const_eval_limit"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("tls_model",
                        &self.tls_model as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "tls_model"));
                }
            }
        };
        {};
        {};
        {
            if (&mut sub_hashes).insert("translate_remapped_path_to_local_path",
                        &self.translate_remapped_path_to_local_path as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "translate_remapped_path_to_local_path"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("trap_unreachable",
                        &self.trap_unreachable as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "trap_unreachable"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("treat_err_as_bug",
                        &self.treat_err_as_bug as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "treat_err_as_bug"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("tune_cpu",
                        &self.tune_cpu as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "tune_cpu"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("typing_mode_post_typeck_until_borrowck",
                        &self.typing_mode_post_typeck_until_borrowck as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "typing_mode_post_typeck_until_borrowck"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("ub_checks",
                        &self.ub_checks as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "ub_checks"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("uninit_const_chunk_threshold",
                        &self.uninit_const_chunk_threshold as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "uninit_const_chunk_threshold"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("unleash_the_miri_inside_of_you",
                        &self.unleash_the_miri_inside_of_you as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "unleash_the_miri_inside_of_you"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("unsound_mir_opts",
                        &self.unsound_mir_opts as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "unsound_mir_opts"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("use_ctors_section",
                        &self.use_ctors_section as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "use_ctors_section"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("use_sync_unwind",
                        &self.use_sync_unwind as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "use_sync_unwind"));
                }
            }
        };
        {};
        {
            if (&mut sub_hashes).insert("verbose_asm",
                        &self.verbose_asm as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "verbose_asm"));
                }
            }
        };
        {
            if !for_crate_hash {
                if (&mut sub_hashes).insert("verbose_internals",
                            &self.verbose_internals as
                                &dyn dep_tracking::DepTrackingHash).is_some() {
                    {
                        ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                                "verbose_internals"));
                    }
                }
            }
        };
        {
            if (&mut sub_hashes).insert("verify_llvm_ir",
                        &self.verify_llvm_ir as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "verify_llvm_ir"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("virtual_function_elimination",
                        &self.virtual_function_elimination as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "virtual_function_elimination"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("wasi_exec_model",
                        &self.wasi_exec_model as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "wasi_exec_model"));
                }
            }
        };
        {
            if (&mut sub_hashes).insert("wasm_c_abi",
                        &self.wasm_c_abi as
                            &dyn dep_tracking::DepTrackingHash).is_some() {
                {
                    ::core::panicking::panic_fmt(format_args!("duplicate key in CLI DepTrackingHash: {0}",
                            "wasm_c_abi"));
                }
            }
        };
        {};
        let mut hasher = StableHasher::new();
        dep_tracking::stable_hash(sub_hashes, &mut hasher, error_format,
            for_crate_hash);
        hasher.finish()
    }
    pub fn gather_target_modifiers(&self, _mods: &mut Vec<TargetModifier>,
        _tmod_vals: &BTreeMap<OptionsTargetModifiers, String>) {
        if self.branch_protection != None {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::BranchProtection),
                _tmod_vals, _mods);
        }
        if self.fixed_x18 != false {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::FixedX18),
                _tmod_vals, _mods);
        }
        if self.indirect_branch_cs_prefix != false {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::IndirectBranchCsPrefix),
                _tmod_vals, _mods);
        }
        if self.reg_struct_return != false {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::RegStructReturn),
                _tmod_vals, _mods);
        }
        if self.regparm != None {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Regparm),
                _tmod_vals, _mods);
        }
        if self.retpoline != false {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Retpoline),
                _tmod_vals, _mods);
        }
        if self.retpoline_external_thunk != false {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::RetpolineExternalThunk),
                _tmod_vals, _mods);
        }
        if self.sanitizer != SanitizerSet::empty() {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Sanitizer),
                _tmod_vals, _mods);
        }
        if self.sanitizer_cfi_normalize_integers != None {
            tmod_push_impl(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers),
                _tmod_vals, _mods);
        }
    }
}
pub const Z_OPTIONS: OptionDescrs<UnstableOptions> =
    &[OptionDesc {
                    name: "allow_features",
                    setter: dbopts::allow_features,
                    type_desc: desc::parse_opt_comma_list,
                    desc: "only allow the listed language features to be enabled in code (comma separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "allow_partial_mitigations",
                    setter: dbopts::allow_partial_mitigations,
                    type_desc: desc::parse_allow_partial_mitigations,
                    desc: "Allow mitigations not enabled for all dependency crates (comma separated list)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "always_encode_mir",
                    setter: dbopts::always_encode_mir,
                    type_desc: desc::parse_bool,
                    desc: "encode MIR of all functions into the crate metadata (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "annotate_moves",
                    setter: dbopts::annotate_moves,
                    type_desc: desc::parse_annotate_moves,
                    desc: "emit debug info for compiler-generated move and copy operations \
        to make them visible in profilers. Can be a boolean or a size limit in bytes (default: disabled)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "assert_incr_state",
                    setter: dbopts::assert_incr_state,
                    type_desc: desc::parse_assert_incr_state,
                    desc: "assert that the incremental cache is in given state: \
         either `loaded` or `not-loaded`.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "assume_incomplete_release",
                    setter: dbopts::assume_incomplete_release,
                    type_desc: desc::parse_bool,
                    desc: "make cfg(version) treat the current version as incomplete (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "assumptions_on_binders",
                    setter: dbopts::assumptions_on_binders,
                    type_desc: desc::parse_bool,
                    desc: "allow deducing higher-ranked outlives assumptions from all binders (`for<'a>`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "autodiff",
                    setter: dbopts::autodiff,
                    type_desc: desc::parse_autodiff,
                    desc: "a list of autodiff flags to enable
        Mandatory setting:
        `=Enable`
        Optional extra settings:
        `=PrintTA`
        `=PrintAA`
        `=PrintPerf`
        `=PrintSteps`
        `=PrintModBefore`
        `=PrintModAfter`
        `=PrintModFinal`
        `=PrintPasses`,
        `=NoPostopt`
        `=LooseTypes`
        `=Inline`
        Multiple options can be combined with commas.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "binary_dep_depinfo",
                    setter: dbopts::binary_dep_depinfo,
                    type_desc: desc::parse_bool,
                    desc: "include artifacts (sysroot, crate dependencies) used during compilation in dep-info \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "box_noalias",
                    setter: dbopts::box_noalias,
                    type_desc: desc::parse_bool,
                    desc: "emit noalias metadata for box (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "branch_protection",
                    setter: dbopts::branch_protection,
                    type_desc: desc::parse_branch_protection,
                    desc: "set options for branch target identification and pointer authentication on AArch64",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::BranchProtection))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "build_sdylib_interface",
                    setter: dbopts::build_sdylib_interface,
                    type_desc: desc::parse_bool,
                    desc: "whether the stable interface is being built",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "cache_proc_macros",
                    setter: dbopts::cache_proc_macros,
                    type_desc: desc::parse_bool,
                    desc: "cache the results of derive proc macro invocations (potentially unsound!) (default: no",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "cf_protection",
                    setter: dbopts::cf_protection,
                    type_desc: desc::parse_cfprotection,
                    desc: "instrument control-flow architecture protection",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "check_cfg_all_expected",
                    setter: dbopts::check_cfg_all_expected,
                    type_desc: desc::parse_bool,
                    desc: "show all expected values in check-cfg diagnostics (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "checksum_hash_algorithm",
                    setter: dbopts::checksum_hash_algorithm,
                    type_desc: desc::parse_cargo_src_file_hash,
                    desc: "hash algorithm of source files used to check freshness in cargo (`blake3` or `sha256`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "codegen_backend",
                    setter: dbopts::codegen_backend,
                    type_desc: desc::parse_opt_string,
                    desc: "the backend to use",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "codegen_emit_retag",
                    setter: dbopts::codegen_emit_retag,
                    type_desc: desc::parse_codegen_retag_options,
                    desc: "emit retag function calls in generated code",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "codegen_source_order",
                    setter: dbopts::codegen_source_order,
                    type_desc: desc::parse_bool,
                    desc: "emit mono items in the order of spans in source files (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "contract_checks",
                    setter: dbopts::contract_checks,
                    type_desc: desc::parse_opt_bool,
                    desc: "emit runtime checks for contract pre- and post-conditions (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "coverage_options",
                    setter: dbopts::coverage_options,
                    type_desc: desc::parse_coverage_options,
                    desc: "control details of coverage instrumentation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "crate_attr",
                    setter: dbopts::crate_attr,
                    type_desc: desc::parse_string_push,
                    desc: "inject the given attribute in the crate",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "cross_crate_inline_threshold",
                    setter: dbopts::cross_crate_inline_threshold,
                    type_desc: desc::parse_inlining_threshold,
                    desc: "threshold to allow cross crate inlining of functions",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "debug_info_type_line_numbers",
                    setter: dbopts::debug_info_type_line_numbers,
                    type_desc: desc::parse_bool,
                    desc: "emit type and line information for additional data types (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "debuginfo_compression",
                    setter: dbopts::debuginfo_compression,
                    type_desc: desc::parse_debuginfo_compression,
                    desc: "compress debug info sections (none, zlib, zstd, default: none)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "debuginfo_for_profiling",
                    setter: dbopts::debuginfo_for_profiling,
                    type_desc: desc::parse_bool,
                    desc: "emit discriminators and other data necessary for AutoFDO",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "deduplicate_diagnostics",
                    setter: dbopts::deduplicate_diagnostics,
                    type_desc: desc::parse_bool,
                    desc: "deduplicate identical diagnostics (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "default_visibility",
                    setter: dbopts::default_visibility,
                    type_desc: desc::parse_opt_symbol_visibility,
                    desc: "overrides the `default_visibility` setting of the target",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "deny_partial_mitigations",
                    setter: dbopts::deny_partial_mitigations,
                    type_desc: desc::parse_deny_partial_mitigations,
                    desc: "Deny mitigations not enabled for all dependency crates (comma separated list)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dep_info_omit_d_target",
                    setter: dbopts::dep_info_omit_d_target,
                    type_desc: desc::parse_bool,
                    desc: "in dep-info output, omit targets for tracking dependencies of the dep-info files \
        themselves (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "direct_access_external_data",
                    setter: dbopts::direct_access_external_data,
                    type_desc: desc::parse_opt_bool,
                    desc: "Direct or use GOT indirect to reference external data symbols",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "disable_fast_paths",
                    setter: dbopts::disable_fast_paths,
                    type_desc: desc::parse_bool,
                    desc: "disable various performance optimizations in trait solving",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "disable_incr_comp_backend_caching",
                    setter: dbopts::disable_incr_comp_backend_caching,
                    type_desc: desc::parse_bool,
                    desc: "disable caching of compiled objects by the codegen backend during incremental compilation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dual_proc_macros",
                    setter: dbopts::dual_proc_macros,
                    type_desc: desc::parse_bool,
                    desc: "load proc macros for both target and host, but only link to the target (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_dep_graph",
                    setter: dbopts::dump_dep_graph,
                    type_desc: desc::parse_bool,
                    desc: "dump the dependency graph to $RUST_DEP_GRAPH (default: /tmp/dep_graph.gv) \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir",
                    setter: dbopts::dump_mir,
                    type_desc: desc::parse_opt_string,
                    desc: "dump MIR state to file.
        `val` is used to select which passes and functions to dump. For example:
        `all` matches all passes and functions,
        `foo` matches all passes for functions whose name contains 'foo',
        `foo & ConstProp` only the 'ConstProp' pass for function names containing 'foo',
        `foo | bar` all passes for function names containing 'foo' or 'bar'.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir_dataflow",
                    setter: dbopts::dump_mir_dataflow,
                    type_desc: desc::parse_bool,
                    desc: "in addition to `.mir` files, create graphviz `.dot` files with dataflow results \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir_dir",
                    setter: dbopts::dump_mir_dir,
                    type_desc: desc::parse_string,
                    desc: "the directory the MIR is dumped into (default: `mir_dump`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir_exclude_alloc_bytes",
                    setter: dbopts::dump_mir_exclude_alloc_bytes,
                    type_desc: desc::parse_bool,
                    desc: "exclude the raw bytes of allocations when dumping MIR (used in tests) (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir_exclude_pass_number",
                    setter: dbopts::dump_mir_exclude_pass_number,
                    type_desc: desc::parse_bool,
                    desc: "exclude the pass number when dumping MIR (used in tests) (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mir_graphviz",
                    setter: dbopts::dump_mir_graphviz,
                    type_desc: desc::parse_bool,
                    desc: "in addition to `.mir` files, create graphviz `.dot` files (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mono_stats",
                    setter: dbopts::dump_mono_stats,
                    type_desc: desc::parse_switch_with_opt_path,
                    desc: "output statistics about monomorphization collection",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dump_mono_stats_format",
                    setter: dbopts::dump_mono_stats_format,
                    type_desc: desc::parse_dump_mono_stats,
                    desc: "the format to use for -Z dump-mono-stats (`markdown` (default) or `json`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dwarf_version",
                    setter: dbopts::dwarf_version,
                    type_desc: desc::parse_opt_number,
                    desc: "version of DWARF debug information to emit (default: 2 or 4, depending on platform)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "dylib_lto",
                    setter: dbopts::dylib_lto,
                    type_desc: desc::parse_bool,
                    desc: "enables LTO for dylib crate type",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "eagerly_emit_delayed_bugs",
                    setter: dbopts::eagerly_emit_delayed_bugs,
                    type_desc: desc::parse_bool,
                    desc: "emit delayed bugs eagerly as errors instead of stashing them and emitting \
        them only if an error has not been emitted",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "ehcont_guard",
                    setter: dbopts::ehcont_guard,
                    type_desc: desc::parse_bool,
                    desc: "generate Windows EHCont Guard tables",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "embed_metadata",
                    setter: dbopts::embed_metadata,
                    type_desc: desc::parse_bool,
                    desc: "embed metadata in rlibs and dylibs (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "embed_source",
                    setter: dbopts::embed_source,
                    type_desc: desc::parse_bool,
                    desc: "embed source text in DWARF debug sections (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "emit_stack_sizes",
                    setter: dbopts::emit_stack_sizes,
                    type_desc: desc::parse_bool,
                    desc: "emit a section containing stack size metadata (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "enforce_type_length_limit",
                    setter: dbopts::enforce_type_length_limit,
                    type_desc: desc::parse_bool,
                    desc: "enforce the type length limit when monomorphizing instances in codegen",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "experimental_default_bounds",
                    setter: dbopts::experimental_default_bounds,
                    type_desc: desc::parse_bool,
                    desc: "enable default bounds for experimental group of auto traits",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "export_executable_symbols",
                    setter: dbopts::export_executable_symbols,
                    type_desc: desc::parse_bool,
                    desc: "export symbols from executables, as if they were dynamic libraries",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "external_clangrt",
                    setter: dbopts::external_clangrt,
                    type_desc: desc::parse_bool,
                    desc: "rely on user specified linker commands to find clangrt",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "extra_const_ub_checks",
                    setter: dbopts::extra_const_ub_checks,
                    type_desc: desc::parse_bool,
                    desc: "turns on more checks to detect const UB, which can be slow (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "fewer_names",
                    setter: dbopts::fewer_names,
                    type_desc: desc::parse_opt_bool,
                    desc: "reduce memory use by retaining fewer names within compilation artifacts (LLVM-IR) \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "fixed_x18",
                    setter: dbopts::fixed_x18,
                    type_desc: desc::parse_bool,
                    desc: "make the x18 register reserved on AArch64 (default: no)",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::FixedX18))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "flatten_format_args",
                    setter: dbopts::flatten_format_args,
                    type_desc: desc::parse_bool,
                    desc: "flatten nested format_args!() and literals into a simplified format_args!() call \
        (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "fmt_debug",
                    setter: dbopts::fmt_debug,
                    type_desc: desc::parse_fmt_debug,
                    desc: "how detailed `#[derive(Debug)]` should be. `full` prints types recursively, \
        `shallow` prints only type names, `none` prints nothing and disables `{:?}`. (default: `full`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "force_unstable_if_unmarked",
                    setter: dbopts::force_unstable_if_unmarked,
                    type_desc: desc::parse_bool,
                    desc: "force all crates to be `rustc_private` unstable (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "function_return",
                    setter: dbopts::function_return,
                    type_desc: desc::parse_function_return,
                    desc: "replace returns with jumps to `__x86_return_thunk` (default: `keep`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "function_sections",
                    setter: dbopts::function_sections,
                    type_desc: desc::parse_opt_bool,
                    desc: "whether each function should go in its own section",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "future_incompat_test",
                    setter: dbopts::future_incompat_test,
                    type_desc: desc::parse_bool,
                    desc: "forces all lints to be future incompatible, used for internal testing (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "graphviz_dark_mode",
                    setter: dbopts::graphviz_dark_mode,
                    type_desc: desc::parse_bool,
                    desc: "use dark-themed colors in graphviz output (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "graphviz_font",
                    setter: dbopts::graphviz_font,
                    type_desc: desc::parse_string,
                    desc: "use the given `fontname` in graphviz output; can be overridden by setting \
        environment variable `RUSTC_GRAPHVIZ_FONT` (default: `Courier, monospace`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "has_thread_local",
                    setter: dbopts::has_thread_local,
                    type_desc: desc::parse_opt_bool,
                    desc: "explicitly enable the `cfg(target_thread_local)` directive",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "help",
                    setter: dbopts::help,
                    type_desc: desc::parse_no_value,
                    desc: "Print unstable compiler options",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "higher_ranked_assumptions",
                    setter: dbopts::higher_ranked_assumptions,
                    type_desc: desc::parse_bool,
                    desc: "allow deducing higher-ranked outlives assumptions from coroutines when proving auto traits",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "hint_mostly_unused",
                    setter: dbopts::hint_mostly_unused,
                    type_desc: desc::parse_bool,
                    desc: "hint that most of this crate will go unused, to minimize work for uncalled functions",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "human_readable_cgu_names",
                    setter: dbopts::human_readable_cgu_names,
                    type_desc: desc::parse_bool,
                    desc: "generate human-readable, predictable names for codegen units (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "identify_regions",
                    setter: dbopts::identify_regions,
                    type_desc: desc::parse_bool,
                    desc: "display unnamed regions as `'<id>`, using a non-ident unique id (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "ignore_directory_in_diagnostics_source_blocks",
                    setter: dbopts::ignore_directory_in_diagnostics_source_blocks,
                    type_desc: desc::parse_string_push,
                    desc: "do not display the source code block in diagnostics for files in the directory",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "incremental_ignore_spans",
                    setter: dbopts::incremental_ignore_spans,
                    type_desc: desc::parse_bool,
                    desc: "ignore spans during ICH computation -- used for testing (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "incremental_info",
                    setter: dbopts::incremental_info,
                    type_desc: desc::parse_bool,
                    desc: "print high-level information about incremental reuse (or the lack thereof) \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "incremental_verify_ich",
                    setter: dbopts::incremental_verify_ich,
                    type_desc: desc::parse_bool,
                    desc: "verify extended properties for incr. comp. (default: no):
        - hashes of green query instances
        - hash collisions of query keys
        - hash collisions when creating dep-nodes",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "indirect_branch_cs_prefix",
                    setter: dbopts::indirect_branch_cs_prefix,
                    type_desc: desc::parse_bool,
                    desc: "add `cs` prefix to `call` and `jmp` to indirect thunks (default: no)",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::IndirectBranchCsPrefix))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_llvm",
                    setter: dbopts::inline_llvm,
                    type_desc: desc::parse_bool,
                    desc: "enable LLVM inlining (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_mir",
                    setter: dbopts::inline_mir,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable MIR inlining (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_mir_forwarder_threshold",
                    setter: dbopts::inline_mir_forwarder_threshold,
                    type_desc: desc::parse_opt_number,
                    desc: "inlining threshold when the caller is a simple forwarding function (default: 30)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_mir_hint_threshold",
                    setter: dbopts::inline_mir_hint_threshold,
                    type_desc: desc::parse_opt_number,
                    desc: "inlining threshold for functions with inline hint (default: 100)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_mir_preserve_debug",
                    setter: dbopts::inline_mir_preserve_debug,
                    type_desc: desc::parse_opt_bool,
                    desc: "when MIR inlining, whether to preserve debug info for callee variables \
        (default: preserve for debuginfo != None, otherwise remove)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "inline_mir_threshold",
                    setter: dbopts::inline_mir_threshold,
                    type_desc: desc::parse_opt_number,
                    desc: "a default MIR inlining threshold (default: 50)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "input_stats",
                    setter: dbopts::input_stats,
                    type_desc: desc::parse_bool,
                    desc: "print some statistics about AST and HIR (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "instrument_mcount",
                    setter: dbopts::instrument_mcount,
                    type_desc: desc::parse_bool,
                    desc: "insert function instrument code for mcount-based tracing (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "instrument_xray",
                    setter: dbopts::instrument_xray,
                    type_desc: desc::parse_instrument_xray,
                    desc: "insert function instrument code for XRay-based tracing (default: no)
         Optional extra settings:
         `=always`
         `=never`
         `=ignore-loops`
         `=instruction-threshold=N`
         `=skip-entry`
         `=skip-exit`
         Multiple options can be combined with commas.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "internal_testing_features",
                    setter: dbopts::internal_testing_features,
                    type_desc: desc::parse_bool,
                    desc: "allow certain internal language features to be enabled that help exercise & test the compiler",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "large_data_threshold",
                    setter: dbopts::large_data_threshold,
                    type_desc: desc::parse_opt_number,
                    desc: "set the threshold for objects to be stored in a \"large data\" section \
         (only effective with -Ccode-model=medium, default: 65536)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "layout_seed",
                    setter: dbopts::layout_seed,
                    type_desc: desc::parse_opt_number,
                    desc: "seed layout randomization",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_directives",
                    setter: dbopts::link_directives,
                    type_desc: desc::parse_bool,
                    desc: "honor #[link] directives in the compiled crate (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_native_libraries",
                    setter: dbopts::link_native_libraries,
                    type_desc: desc::parse_bool,
                    desc: "link native libraries in the linker invocation (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "link_only",
                    setter: dbopts::link_only,
                    type_desc: desc::parse_bool,
                    desc: "link the `.rlink` file generated by `-Z no-link` (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "lint_llvm_ir",
                    setter: dbopts::lint_llvm_ir,
                    type_desc: desc::parse_bool,
                    desc: "lint LLVM IR (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "lint_mir",
                    setter: dbopts::lint_mir,
                    type_desc: desc::parse_bool,
                    desc: "lint MIR before and after each transformation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "lint_rust_version",
                    setter: dbopts::lint_rust_version,
                    type_desc: desc::parse_rust_version,
                    desc: "control the minimum rust version for lints",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "llvm_module_flag",
                    setter: dbopts::llvm_module_flag,
                    type_desc: desc::parse_llvm_module_flag,
                    desc: "a list of module flags to pass to LLVM (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "llvm_plugins",
                    setter: dbopts::llvm_plugins,
                    type_desc: desc::parse_list,
                    desc: "a list LLVM plugins to enable (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "llvm_time_trace",
                    setter: dbopts::llvm_time_trace,
                    type_desc: desc::parse_bool,
                    desc: "generate JSON tracing data file from LLVM data (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "llvm_writable",
                    setter: dbopts::llvm_writable,
                    type_desc: desc::parse_bool,
                    desc: "emit the LLVM writable attribute for mutable reference arguments (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "location_detail",
                    setter: dbopts::location_detail,
                    type_desc: desc::parse_location_detail,
                    desc: "what location details should be tracked when using caller_location, either \
        `none`, or a comma separated list of location details, for which \
        valid options are `file`, `line`, and `column` (default: `file,line,column`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "ls",
                    setter: dbopts::ls,
                    type_desc: desc::parse_list,
                    desc: "decode and print various parts of the crate metadata for a library crate \
        (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "macro_backtrace",
                    setter: dbopts::macro_backtrace,
                    type_desc: desc::parse_bool,
                    desc: "show macro backtraces (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "macro_stats",
                    setter: dbopts::macro_stats,
                    type_desc: desc::parse_bool,
                    desc: "print some statistics about macro expansions (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "maximal_hir_to_mir_coverage",
                    setter: dbopts::maximal_hir_to_mir_coverage,
                    type_desc: desc::parse_bool,
                    desc: "save as much information as possible about the correspondence between MIR and HIR \
        as source scopes (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "merge_functions",
                    setter: dbopts::merge_functions,
                    type_desc: desc::parse_merge_functions,
                    desc: "control the operation of the MergeFunctions LLVM pass, taking \
        the same values as the target option of the same name",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "meta_stats",
                    setter: dbopts::meta_stats,
                    type_desc: desc::parse_bool,
                    desc: "gather metadata statistics (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "metrics_dir",
                    setter: dbopts::metrics_dir,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "the directory metrics emitted by rustc are dumped into (implicitly enables default set of metrics)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "min_function_alignment",
                    setter: dbopts::min_function_alignment,
                    type_desc: desc::parse_align,
                    desc: "align all functions to at least this many bytes. Must be a power of 2",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "min_recursion_limit",
                    setter: dbopts::min_recursion_limit,
                    type_desc: desc::parse_opt_number,
                    desc: "set a minimum recursion limit (final limit = max(this, recursion_limit_from_crate))",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_enable_passes",
                    setter: dbopts::mir_enable_passes,
                    type_desc: desc::parse_list_with_polarity,
                    desc: "use like `-Zmir-enable-passes=+DestinationPropagation,-InstSimplify`. Forces the \
        specified passes to be enabled, overriding all other checks. In particular, this will \
        enable unsound (known-buggy and hence usually disabled) passes without further warning! \
        Passes that are not specified are enabled or disabled by other flags as usual.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_include_spans",
                    setter: dbopts::mir_include_spans,
                    type_desc: desc::parse_mir_include_spans,
                    desc: "include extra comments in mir pretty printing, like line numbers and statement indices, \
         details about types, etc. (boolean for all passes, 'nll' to enable in NLL MIR only, default: 'nll')",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_opt_bisect_limit",
                    setter: dbopts::mir_opt_bisect_limit,
                    type_desc: desc::parse_opt_number,
                    desc: "limit the number of MIR optimization pass executions (global across all bodies). \
        Pass executions after this limit are skipped and reported. (default: no limit)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_opt_level",
                    setter: dbopts::mir_opt_level,
                    type_desc: desc::parse_opt_number,
                    desc: "MIR optimization level (0-4; default: 1 in non optimized builds and 2 in optimized builds)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_preserve_ub",
                    setter: dbopts::mir_preserve_ub,
                    type_desc: desc::parse_bool,
                    desc: "keep place mention statements and reads in trivial SwitchInt terminators, which are interpreted \
        e.g., by miri; implies -Zmir-opt-level=0 (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mir_strip_debuginfo",
                    setter: dbopts::mir_strip_debuginfo,
                    type_desc: desc::parse_mir_strip_debuginfo,
                    desc: "Whether to remove some of the MIR debug info from methods.  Default: None",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "move_size_limit",
                    setter: dbopts::move_size_limit,
                    type_desc: desc::parse_opt_number,
                    desc: "the size at which the `large_assignments` lint starts to be emitted",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "mutable_noalias",
                    setter: dbopts::mutable_noalias,
                    type_desc: desc::parse_bool,
                    desc: "emit noalias metadata for mutable references (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "namespaced_crates",
                    setter: dbopts::namespaced_crates,
                    type_desc: desc::parse_bool,
                    desc: "allow crates to be namespaced by other crates (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "next_solver",
                    setter: dbopts::next_solver,
                    type_desc: desc::parse_next_solver_config,
                    desc: "enable and configure the next generation trait solver used by rustc",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "nll_facts",
                    setter: dbopts::nll_facts,
                    type_desc: desc::parse_bool,
                    desc: "dump facts from NLL analysis into side files (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "nll_facts_dir",
                    setter: dbopts::nll_facts_dir,
                    type_desc: desc::parse_string,
                    desc: "the directory the NLL facts are dumped into (default: `nll-facts`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_analysis",
                    setter: dbopts::no_analysis,
                    type_desc: desc::parse_no_value,
                    desc: "parse and expand the source, but run no analysis",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_codegen",
                    setter: dbopts::no_codegen,
                    type_desc: desc::parse_no_value,
                    desc: "run all passes except codegen; no output",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_generate_arange_section",
                    setter: dbopts::no_generate_arange_section,
                    type_desc: desc::parse_no_value,
                    desc: "omit DWARF address ranges that give faster lookups",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_implied_bounds_compat",
                    setter: dbopts::no_implied_bounds_compat,
                    type_desc: desc::parse_bool,
                    desc: "disable the compatibility version of the `implied_bounds_ty` query",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_leak_check",
                    setter: dbopts::no_leak_check,
                    type_desc: desc::parse_no_value,
                    desc: "disable the 'leak check' for subtyping; unsound, but useful for tests",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_link",
                    setter: dbopts::no_link,
                    type_desc: desc::parse_no_value,
                    desc: "compile without linking",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_parallel_backend",
                    setter: dbopts::no_parallel_backend,
                    type_desc: desc::parse_no_value,
                    desc: "run LLVM in non-parallel mode (while keeping codegen-units and ThinLTO)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_profiler_runtime",
                    setter: dbopts::no_profiler_runtime,
                    type_desc: desc::parse_no_value,
                    desc: "prevent automatic injection of the profiler_builtins crate",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_steal_thir",
                    setter: dbopts::no_steal_thir,
                    type_desc: desc::parse_bool,
                    desc: "don't steal the THIR when we're done with it; useful for rustc drivers (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_trait_vptr",
                    setter: dbopts::no_trait_vptr,
                    type_desc: desc::parse_no_value,
                    desc: "disable generation of trait vptr in vtable for upcasting",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "no_unique_section_names",
                    setter: dbopts::no_unique_section_names,
                    type_desc: desc::parse_bool,
                    desc: "do not use unique names for text and data sections when -Z function-sections is used",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "normalize_docs",
                    setter: dbopts::normalize_docs,
                    type_desc: desc::parse_bool,
                    desc: "normalize associated items in rustdoc when generating documentation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "offload",
                    setter: dbopts::offload,
                    type_desc: desc::parse_offload,
                    desc: "a list of offload flags to enable
        Mandatory setting:
        `=Enable`
        Currently the only option available",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "on_broken_pipe",
                    setter: dbopts::on_broken_pipe,
                    type_desc: desc::parse_on_broken_pipe,
                    desc: "behavior of std::io::ErrorKind::BrokenPipe (SIGPIPE)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "osx_rpath_install_name",
                    setter: dbopts::osx_rpath_install_name,
                    type_desc: desc::parse_bool,
                    desc: "pass `-install_name @rpath/...` to the macOS linker (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "packed_bundled_libs",
                    setter: dbopts::packed_bundled_libs,
                    type_desc: desc::parse_bool,
                    desc: "change rlib format to store native libraries as archives",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "packed_stack",
                    setter: dbopts::packed_stack,
                    type_desc: desc::parse_bool,
                    desc: "use packed stack frames (s390x only) (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "panic_abort_tests",
                    setter: dbopts::panic_abort_tests,
                    type_desc: desc::parse_bool,
                    desc: "support compiling tests with panic=abort (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "panic_in_drop",
                    setter: dbopts::panic_in_drop,
                    type_desc: desc::parse_panic_strategy,
                    desc: "panic strategy for panics in drops",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "parse_crate_root_only",
                    setter: dbopts::parse_crate_root_only,
                    type_desc: desc::parse_bool,
                    desc: "parse the crate root file only; do not parse other files, compile, assemble, or link \
        (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "patchable_function_entry",
                    setter: dbopts::patchable_function_entry,
                    type_desc: desc::parse_patchable_function_entry,
                    desc: "nop padding at function entry",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "plt",
                    setter: dbopts::plt,
                    type_desc: desc::parse_opt_bool,
                    desc: "whether to use the PLT when calling into shared libraries;
        only has effect for PIC code on systems with ELF binaries
        (default: PLT is disabled if full relro is enabled on x86_64)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "polonius",
                    setter: dbopts::polonius,
                    type_desc: desc::parse_polonius,
                    desc: "enable polonius-based borrow-checker (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "pre_link_arg",
                    setter: dbopts::pre_link_arg,
                    type_desc: desc::parse_string_push,
                    desc: "a single extra argument to prepend the linker invocation (can be used several times)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "pre_link_args",
                    setter: dbopts::pre_link_args,
                    type_desc: desc::parse_list,
                    desc: "extra arguments to prepend to the linker invocation (space separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "precise_enum_drop_elaboration",
                    setter: dbopts::precise_enum_drop_elaboration,
                    type_desc: desc::parse_bool,
                    desc: "use a more precise version of drop elaboration for matches on enums (default: yes). \
        This results in better codegen, but has caused miscompilations on some tier 2 platforms. \
        See #77382 and #74551.",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "print_codegen_stats",
                    setter: dbopts::print_codegen_stats,
                    type_desc: desc::parse_bool,
                    desc: "print codegen statistics (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "print_codegen_stats_json",
                    setter: dbopts::print_codegen_stats_json,
                    type_desc: desc::parse_opt_string,
                    desc: "print codegen statistics in JSON to a file (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "print_llvm_passes",
                    setter: dbopts::print_llvm_passes,
                    type_desc: desc::parse_bool,
                    desc: "print the LLVM optimization passes being run (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "print_mono_items",
                    setter: dbopts::print_mono_items,
                    type_desc: desc::parse_bool,
                    desc: "print the result of the monomorphization collection pass (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "print_type_sizes",
                    setter: dbopts::print_type_sizes,
                    type_desc: desc::parse_bool,
                    desc: "print layout information for each type encountered (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "proc_macro_backtrace",
                    setter: dbopts::proc_macro_backtrace,
                    type_desc: desc::parse_bool,
                    desc: "show backtraces for panics during proc-macro execution (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "proc_macro_execution_strategy",
                    setter: dbopts::proc_macro_execution_strategy,
                    type_desc: desc::parse_proc_macro_execution_strategy,
                    desc: "how to run proc-macro code (default: same-thread)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "profile_closures",
                    setter: dbopts::profile_closures,
                    type_desc: desc::parse_no_value,
                    desc: "profile size of closures",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "profile_sample_use",
                    setter: dbopts::profile_sample_use,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "use the given `.prof` file for sampled profile-guided optimization (also known as AutoFDO)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "profiler_runtime",
                    setter: dbopts::profiler_runtime,
                    type_desc: desc::parse_string,
                    desc: "name of the profiler runtime crate to automatically inject (default: `profiler_builtins`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "query_dep_graph",
                    setter: dbopts::query_dep_graph,
                    type_desc: desc::parse_bool,
                    desc: "enable queries of the dependency graph for regression testing (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "randomize_layout",
                    setter: dbopts::randomize_layout,
                    type_desc: desc::parse_bool,
                    desc: "randomize the layout of types (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "reg_struct_return",
                    setter: dbopts::reg_struct_return,
                    type_desc: desc::parse_bool,
                    desc: "On x86-32 targets, it overrides the default ABI to return small structs in registers.
        It is UNSOUND to link together crates that use different values for this flag!",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::RegStructReturn))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "regparm",
                    setter: dbopts::regparm,
                    type_desc: desc::parse_opt_number,
                    desc: "On x86-32 targets, setting this to N causes the compiler to pass N arguments \
        in registers EAX, EDX, and ECX instead of on the stack for\
        \"C\", \"cdecl\", and \"stdcall\" fn.\
        It is UNSOUND to link together crates that use different values for this flag!",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Regparm))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "relax_elf_relocations",
                    setter: dbopts::relax_elf_relocations,
                    type_desc: desc::parse_opt_bool,
                    desc: "whether ELF relocations can be relaxed",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "remap_cwd_prefix",
                    setter: dbopts::remap_cwd_prefix,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "remap paths under the current working directory to this path prefix",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "remark_dir",
                    setter: dbopts::remark_dir,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "directory into which to write optimization remarks (if not specified, they will be \
written to standard error output)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "retpoline",
                    setter: dbopts::retpoline,
                    type_desc: desc::parse_bool,
                    desc: "enables retpoline-indirect-branches and retpoline-indirect-calls target features (default: no)",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Retpoline))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "retpoline_external_thunk",
                    setter: dbopts::retpoline_external_thunk,
                    type_desc: desc::parse_bool,
                    desc: "enables retpoline-external-thunk, retpoline-indirect-branches and retpoline-indirect-calls \
        target features (default: no)",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::RetpolineExternalThunk))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer",
                    setter: dbopts::sanitizer,
                    type_desc: desc::parse_sanitizers,
                    desc: "use a sanitizer",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::Sanitizer))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_cfi_canonical_jump_tables",
                    setter: dbopts::sanitizer_cfi_canonical_jump_tables,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable canonical jump tables (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_cfi_generalize_pointers",
                    setter: dbopts::sanitizer_cfi_generalize_pointers,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable generalizing pointer types (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_cfi_normalize_integers",
                    setter: dbopts::sanitizer_cfi_normalize_integers,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable normalizing integer types (default: no)",
                    removed: None,
                    tmod: None.or(Some(OptionsTargetModifiers::UnstableOptions(UnstableOptionsTargetModifiers::SanitizerCfiNormalizeIntegers))),
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_dataflow_abilist",
                    setter: dbopts::sanitizer_dataflow_abilist,
                    type_desc: desc::parse_comma_list,
                    desc: "additional ABI list files that control how shadow parameters are passed (comma separated)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_kcfi_arity",
                    setter: dbopts::sanitizer_kcfi_arity,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable KCFI arity indicator (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_memory_track_origins",
                    setter: dbopts::sanitizer_memory_track_origins,
                    type_desc: desc::parse_sanitizer_memory_track_origins,
                    desc: "enable origins tracking in MemorySanitizer",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "sanitizer_recover",
                    setter: dbopts::sanitizer_recover,
                    type_desc: desc::parse_sanitizers,
                    desc: "enable recovery for selected sanitizers",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "saturating_float_casts",
                    setter: dbopts::saturating_float_casts,
                    type_desc: desc::parse_opt_bool,
                    desc: "make float->int casts UB-free: numbers outside the integer type's range are clipped to \
        the max/min integer respectively, and NaN is mapped to 0 (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "self_profile",
                    setter: dbopts::self_profile,
                    type_desc: desc::parse_switch_with_opt_path,
                    desc: "run the self profiler and output the raw event data",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "self_profile_counter",
                    setter: dbopts::self_profile_counter,
                    type_desc: desc::parse_string,
                    desc: "counter used by the self profiler (default: `wall-time`), one of:
        `wall-time` (monotonic clock, i.e. `std::time::Instant`)
        `instructions:u` (retired instructions, userspace-only)
        `instructions-minus-irqs:u` (subtracting hardware interrupt counts for extra accuracy)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "self_profile_events",
                    setter: dbopts::self_profile_events,
                    type_desc: desc::parse_opt_comma_list,
                    desc: "specify the events recorded by the self profiler;
        for example: `-Z self-profile-events=default,query-keys`
        all options: none, all, default, generic-activity, query-provider, query-cache-hit
                     query-blocked, incr-cache-load, incr-result-hashing, query-keys, function-args, args, llvm, artifact-sizes",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "share_generics",
                    setter: dbopts::share_generics,
                    type_desc: desc::parse_opt_bool,
                    desc: "make the current crate share its generic instantiations",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "shell_argfiles",
                    setter: dbopts::shell_argfiles,
                    type_desc: desc::parse_bool,
                    desc: "allow argument files to be specified with POSIX \"shell-style\" argument quoting",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "simulate_remapped_rust_src_base",
                    setter: dbopts::simulate_remapped_rust_src_base,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "simulate the effect of remap-debuginfo = true at bootstrapping by remapping path \
        to rust's source base directory. only meant for testing purposes",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "small_data_threshold",
                    setter: dbopts::small_data_threshold,
                    type_desc: desc::parse_opt_number,
                    desc: "Set the threshold for objects to be stored in a \"small data\" section",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "span_debug",
                    setter: dbopts::span_debug,
                    type_desc: desc::parse_bool,
                    desc: "forward proc_macro::Span's `Debug` impl to `Span`",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "span_free_formats",
                    setter: dbopts::span_free_formats,
                    type_desc: desc::parse_bool,
                    desc: "exclude spans when debug-printing compiler state (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "split_dwarf_inlining",
                    setter: dbopts::split_dwarf_inlining,
                    type_desc: desc::parse_bool,
                    desc: "provide minimal debug info in the object/executable to facilitate online \
         symbolication/stack traces in the absence of .dwo/.dwp files when using Split DWARF",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "split_dwarf_kind",
                    setter: dbopts::split_dwarf_kind,
                    type_desc: desc::parse_split_dwarf_kind,
                    desc: "split dwarf variant (only if -Csplit-debuginfo is enabled and on relevant platform)
        (default: `split`)

        `split`: sections which do not require relocation are written into a DWARF object (`.dwo`)
                 file which is ignored by the linker
        `single`: sections which do not require relocation are written into object file but ignored
                  by the linker",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "split_dwarf_out_dir",
                    setter: dbopts::split_dwarf_out_dir,
                    type_desc: desc::parse_opt_pathbuf,
                    desc: "location for writing split DWARF objects (`.dwo`) if enabled",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "split_lto_unit",
                    setter: dbopts::split_lto_unit,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable LTO unit splitting (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "src_hash_algorithm",
                    setter: dbopts::src_hash_algorithm,
                    type_desc: desc::parse_src_file_hash,
                    desc: "hash algorithm of source files in debug info (`md5`, `sha1`, or `sha256`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "stack_protector",
                    setter: dbopts::stack_protector,
                    type_desc: desc::parse_stack_protector,
                    desc: "control stack smash protection strategy (`rustc --print stack-protector-strategies` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None.or(Some(mitigation_coverage::DeniedPartialMitigationKind::StackProtector)),
                },
                OptionDesc {
                    name: "staticlib_allow_rdylib_deps",
                    setter: dbopts::staticlib_allow_rdylib_deps,
                    type_desc: desc::parse_bool,
                    desc: "allow staticlibs to have rust dylib dependencies",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "staticlib_hide_internal_symbols",
                    setter: dbopts::staticlib_hide_internal_symbols,
                    type_desc: desc::parse_bool,
                    desc: "hide non-exported Rust symbols when building staticlibs by setting STV_HIDDEN",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "staticlib_rename_internal_symbols",
                    setter: dbopts::staticlib_rename_internal_symbols,
                    type_desc: desc::parse_bool,
                    desc: "rename non-exported Rust symbols when building staticlibs to avoid conflicts",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "staticlib_prefer_dynamic",
                    setter: dbopts::staticlib_prefer_dynamic,
                    type_desc: desc::parse_bool,
                    desc: "prefer dynamic linking to static linking for staticlibs (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "strict_init_checks",
                    setter: dbopts::strict_init_checks,
                    type_desc: desc::parse_bool,
                    desc: "control if mem::uninitialized and mem::zeroed panic on more UB",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "teach",
                    setter: dbopts::teach,
                    type_desc: desc::parse_bool,
                    desc: "show extended diagnostic help (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "temps_dir",
                    setter: dbopts::temps_dir,
                    type_desc: desc::parse_opt_string,
                    desc: "the directory the intermediate files are written to",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "terminal_urls",
                    setter: dbopts::terminal_urls,
                    type_desc: desc::parse_terminal_url,
                    desc: "use the OSC 8 hyperlink terminal specification to print hyperlinks in the compiler output",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "thinlto",
                    setter: dbopts::thinlto,
                    type_desc: desc::parse_opt_bool,
                    desc: "enable ThinLTO when possible",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "threads",
                    setter: dbopts::threads,
                    type_desc: desc::parse_threads,
                    desc: "use a thread pool with N threads",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "time_llvm_passes",
                    setter: dbopts::time_llvm_passes,
                    type_desc: desc::parse_bool,
                    desc: "measure time of each LLVM pass (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "time_passes",
                    setter: dbopts::time_passes,
                    type_desc: desc::parse_bool,
                    desc: "measure time of each rustc pass (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "time_passes_format",
                    setter: dbopts::time_passes_format,
                    type_desc: desc::parse_time_passes_format,
                    desc: "the format to use for -Z time-passes (`text` (default) or `json`)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "tiny_const_eval_limit",
                    setter: dbopts::tiny_const_eval_limit,
                    type_desc: desc::parse_bool,
                    desc: "sets a tiny, non-configurable limit for const eval; useful for compiler tests",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "tls_model",
                    setter: dbopts::tls_model,
                    type_desc: desc::parse_tls_model,
                    desc: "choose the TLS model to use (`rustc --print tls-models` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "trace_macros",
                    setter: dbopts::trace_macros,
                    type_desc: desc::parse_bool,
                    desc: "for every macro invocation, print its name and arguments (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "track_diagnostics",
                    setter: dbopts::track_diagnostics,
                    type_desc: desc::parse_bool,
                    desc: "tracks where in rustc a diagnostic was emitted",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "translate_remapped_path_to_local_path",
                    setter: dbopts::translate_remapped_path_to_local_path,
                    type_desc: desc::parse_bool,
                    desc: "translate remapped paths into local paths when possible (default: yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "trap_unreachable",
                    setter: dbopts::trap_unreachable,
                    type_desc: desc::parse_opt_bool,
                    desc: "generate trap instructions for unreachable intrinsics (default: use target setting, usually yes)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "treat_err_as_bug",
                    setter: dbopts::treat_err_as_bug,
                    type_desc: desc::parse_treat_err_as_bug,
                    desc: "treat the `val`th error that occurs as bug (default if not specified: 0 - don't treat errors as bugs. \
        default if specified without a value: 1 - treat the first error as bug)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "trim_diagnostic_paths",
                    setter: dbopts::trim_diagnostic_paths,
                    type_desc: desc::parse_bool,
                    desc: "in diagnostics, use heuristics to shorten paths referring to items",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "tune_cpu",
                    setter: dbopts::tune_cpu,
                    type_desc: desc::parse_opt_string,
                    desc: "select processor to schedule for (`rustc --print target-cpus` for details)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "typing_mode_post_typeck_until_borrowck",
                    setter: dbopts::typing_mode_post_typeck_until_borrowck,
                    type_desc: desc::parse_bool,
                    desc: "enable `TypingMode::PostTypeckUntilBorrowck`, changing the way opaque types are handled during MIR borrowck",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "ub_checks",
                    setter: dbopts::ub_checks,
                    type_desc: desc::parse_opt_bool,
                    desc: "emit runtime checks for Undefined Behavior (default: -Cdebug-assertions)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "ui_testing",
                    setter: dbopts::ui_testing,
                    type_desc: desc::parse_bool,
                    desc: "emit compiler diagnostics in a form suitable for UI testing (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "uninit_const_chunk_threshold",
                    setter: dbopts::uninit_const_chunk_threshold,
                    type_desc: desc::parse_number,
                    desc: "allow generating const initializers with mixed init/uninit chunks, \
        and set the maximum number of chunks for which this is allowed (default: 16)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "unleash_the_miri_inside_of_you",
                    setter: dbopts::unleash_the_miri_inside_of_you,
                    type_desc: desc::parse_bool,
                    desc: "take the brakes off const evaluation. NOTE: this is unsound (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "unpretty",
                    setter: dbopts::unpretty,
                    type_desc: desc::parse_unpretty,
                    desc: "present the input source, unstable (and less-pretty) variants;
        `normal`,
        `expanded`, `expanded,identified`,
        `expanded,hygiene` (with internal representations),
        `ast-tree` (raw AST before expansion),
        `ast-tree,expanded` (raw AST after expansion),
        `hir` (the HIR), `hir,identified`,
        `hir,typed` (HIR with types for each node),
        `hir-tree` (dump the raw HIR),
        `thir-tree`, `thir-flat`,
        `mir` (the MIR), or `mir-cfg` (graphviz formatted MIR)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "unsound_mir_opts",
                    setter: dbopts::unsound_mir_opts,
                    type_desc: desc::parse_bool,
                    desc: "enable unsound and buggy MIR optimizations (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "unstable_options",
                    setter: dbopts::unstable_options,
                    type_desc: desc::parse_no_value,
                    desc: "adds unstable command line options to rustc interface (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "use_ctors_section",
                    setter: dbopts::use_ctors_section,
                    type_desc: desc::parse_opt_bool,
                    desc: "use legacy .ctors section for initializers rather than .init_array",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "use_sync_unwind",
                    setter: dbopts::use_sync_unwind,
                    type_desc: desc::parse_opt_bool,
                    desc: "Generate sync unwind tables instead of async unwind tables (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "validate_mir",
                    setter: dbopts::validate_mir,
                    type_desc: desc::parse_bool,
                    desc: "validate MIR after each transformation",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "verbose_asm",
                    setter: dbopts::verbose_asm,
                    type_desc: desc::parse_bool,
                    desc: "add descriptive comments from LLVM to the assembly (may change behavior) (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "verbose_internals",
                    setter: dbopts::verbose_internals,
                    type_desc: desc::parse_bool,
                    desc: "in general, enable more debug printouts (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "verify_llvm_ir",
                    setter: dbopts::verify_llvm_ir,
                    type_desc: desc::parse_bool,
                    desc: "verify LLVM IR (default: no)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "virtual_function_elimination",
                    setter: dbopts::virtual_function_elimination,
                    type_desc: desc::parse_bool,
                    desc: "enables dead virtual function elimination optimization. \
        Requires `-Clto[=[fat,yes]]`",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "wasi_exec_model",
                    setter: dbopts::wasi_exec_model,
                    type_desc: desc::parse_wasi_exec_model,
                    desc: "whether to build a wasi command or reactor",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "wasm_c_abi",
                    setter: dbopts::wasm_c_abi,
                    type_desc: desc::parse_wasm_c_abi,
                    desc: "use spec-compliant C ABI for `wasm32-unknown-unknown` (deprecated, always enabled)",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                },
                OptionDesc {
                    name: "write_long_types_to_disk",
                    setter: dbopts::write_long_types_to_disk,
                    type_desc: desc::parse_bool,
                    desc: "whether long type names should be written to files instead of being printed in errors",
                    removed: None,
                    tmod: None,
                    mitigation: None,
                }];
mod dbopts {
    pub(super) fn allow_features(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_comma_list(&mut cg.allow_features, v)
    }
    pub(super) fn allow_partial_mitigations(_cg: &mut super::UnstableOptions,
        collected: &mut super::CollectedOptions, v: Option<&str>,
        index: usize) -> bool {
        collected.mitigations.handle_allowdeny_mitigation_option(v, index,
            true)
    }
    pub(super) fn always_encode_mir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.always_encode_mir, v)
    }
    pub(super) fn annotate_moves(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_annotate_moves(&mut cg.annotate_moves, v)
    }
    pub(super) fn assert_incr_state(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_assert_incr_state(&mut cg.assert_incr_state, v)
    }
    pub(super) fn assume_incomplete_release(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.assume_incomplete_release, v)
    }
    pub(super) fn assumptions_on_binders(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.assumptions_on_binders, v)
    }
    pub(super) fn autodiff(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_autodiff(&mut cg.autodiff, v)
    }
    pub(super) fn binary_dep_depinfo(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.binary_dep_depinfo, v)
    }
    pub(super) fn box_noalias(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.box_noalias, v)
    }
    pub(super) fn branch_protection(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_branch_protection(&mut cg.branch_protection, v)
    }
    pub(super) fn build_sdylib_interface(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.build_sdylib_interface, v)
    }
    pub(super) fn cache_proc_macros(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.cache_proc_macros, v)
    }
    pub(super) fn cf_protection(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_cfprotection(&mut cg.cf_protection, v)
    }
    pub(super) fn check_cfg_all_expected(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.check_cfg_all_expected, v)
    }
    pub(super) fn checksum_hash_algorithm(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_cargo_src_file_hash(&mut cg.checksum_hash_algorithm,
            v)
    }
    pub(super) fn codegen_backend(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.codegen_backend, v)
    }
    pub(super) fn codegen_emit_retag(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_codegen_retag_options(&mut cg.codegen_emit_retag,
            v)
    }
    pub(super) fn codegen_source_order(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.codegen_source_order, v)
    }
    pub(super) fn contract_checks(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.contract_checks, v)
    }
    pub(super) fn coverage_options(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_coverage_options(&mut cg.coverage_options, v)
    }
    pub(super) fn crate_attr(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string_push(&mut cg.crate_attr, v)
    }
    pub(super) fn cross_crate_inline_threshold(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_inlining_threshold(&mut cg.cross_crate_inline_threshold,
            v)
    }
    pub(super) fn debug_info_type_line_numbers(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.debug_info_type_line_numbers, v)
    }
    pub(super) fn debuginfo_compression(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_debuginfo_compression(&mut cg.debuginfo_compression,
            v)
    }
    pub(super) fn debuginfo_for_profiling(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.debuginfo_for_profiling, v)
    }
    pub(super) fn deduplicate_diagnostics(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.deduplicate_diagnostics, v)
    }
    pub(super) fn default_visibility(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_symbol_visibility(&mut cg.default_visibility,
            v)
    }
    pub(super) fn deny_partial_mitigations(_cg: &mut super::UnstableOptions,
        collected: &mut super::CollectedOptions, v: Option<&str>,
        index: usize) -> bool {
        collected.mitigations.handle_allowdeny_mitigation_option(v, index,
            false)
    }
    pub(super) fn dep_info_omit_d_target(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dep_info_omit_d_target, v)
    }
    pub(super) fn direct_access_external_data(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.direct_access_external_data, v)
    }
    pub(super) fn disable_fast_paths(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.disable_fast_paths, v)
    }
    pub(super) fn disable_incr_comp_backend_caching(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.disable_incr_comp_backend_caching, v)
    }
    pub(super) fn dual_proc_macros(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dual_proc_macros, v)
    }
    pub(super) fn dump_dep_graph(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dump_dep_graph, v)
    }
    pub(super) fn dump_mir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.dump_mir, v)
    }
    pub(super) fn dump_mir_dataflow(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dump_mir_dataflow, v)
    }
    pub(super) fn dump_mir_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.dump_mir_dir, v)
    }
    pub(super) fn dump_mir_exclude_alloc_bytes(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dump_mir_exclude_alloc_bytes, v)
    }
    pub(super) fn dump_mir_exclude_pass_number(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dump_mir_exclude_pass_number, v)
    }
    pub(super) fn dump_mir_graphviz(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dump_mir_graphviz, v)
    }
    pub(super) fn dump_mono_stats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_switch_with_opt_path(&mut cg.dump_mono_stats, v)
    }
    pub(super) fn dump_mono_stats_format(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_dump_mono_stats(&mut cg.dump_mono_stats_format, v)
    }
    pub(super) fn dwarf_version(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.dwarf_version, v)
    }
    pub(super) fn dylib_lto(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.dylib_lto, v)
    }
    pub(super) fn eagerly_emit_delayed_bugs(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.eagerly_emit_delayed_bugs, v)
    }
    pub(super) fn ehcont_guard(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.ehcont_guard, v)
    }
    pub(super) fn embed_metadata(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.embed_metadata, v)
    }
    pub(super) fn embed_source(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.embed_source, v)
    }
    pub(super) fn emit_stack_sizes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.emit_stack_sizes, v)
    }
    pub(super) fn enforce_type_length_limit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.enforce_type_length_limit, v)
    }
    pub(super) fn experimental_default_bounds(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.experimental_default_bounds, v)
    }
    pub(super) fn export_executable_symbols(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.export_executable_symbols, v)
    }
    pub(super) fn external_clangrt(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.external_clangrt, v)
    }
    pub(super) fn extra_const_ub_checks(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.extra_const_ub_checks, v)
    }
    pub(super) fn fewer_names(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.fewer_names, v)
    }
    pub(super) fn fixed_x18(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.fixed_x18, v)
    }
    pub(super) fn flatten_format_args(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.flatten_format_args, v)
    }
    pub(super) fn fmt_debug(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_fmt_debug(&mut cg.fmt_debug, v)
    }
    pub(super) fn force_unstable_if_unmarked(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.force_unstable_if_unmarked, v)
    }
    pub(super) fn function_return(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_function_return(&mut cg.function_return, v)
    }
    pub(super) fn function_sections(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.function_sections, v)
    }
    pub(super) fn future_incompat_test(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.future_incompat_test, v)
    }
    pub(super) fn graphviz_dark_mode(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.graphviz_dark_mode, v)
    }
    pub(super) fn graphviz_font(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.graphviz_font, v)
    }
    pub(super) fn has_thread_local(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.has_thread_local, v)
    }
    pub(super) fn help(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.help, v)
    }
    pub(super) fn higher_ranked_assumptions(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.higher_ranked_assumptions, v)
    }
    pub(super) fn hint_mostly_unused(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.hint_mostly_unused, v)
    }
    pub(super) fn human_readable_cgu_names(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.human_readable_cgu_names, v)
    }
    pub(super) fn identify_regions(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.identify_regions, v)
    }
    pub(super) fn ignore_directory_in_diagnostics_source_blocks(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string_push(&mut cg.ignore_directory_in_diagnostics_source_blocks,
            v)
    }
    pub(super) fn incremental_ignore_spans(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.incremental_ignore_spans, v)
    }
    pub(super) fn incremental_info(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.incremental_info, v)
    }
    pub(super) fn incremental_verify_ich(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.incremental_verify_ich, v)
    }
    pub(super) fn indirect_branch_cs_prefix(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.indirect_branch_cs_prefix, v)
    }
    pub(super) fn inline_llvm(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.inline_llvm, v)
    }
    pub(super) fn inline_mir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.inline_mir, v)
    }
    pub(super) fn inline_mir_forwarder_threshold(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.inline_mir_forwarder_threshold,
            v)
    }
    pub(super) fn inline_mir_hint_threshold(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.inline_mir_hint_threshold, v)
    }
    pub(super) fn inline_mir_preserve_debug(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.inline_mir_preserve_debug, v)
    }
    pub(super) fn inline_mir_threshold(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.inline_mir_threshold, v)
    }
    pub(super) fn input_stats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.input_stats, v)
    }
    pub(super) fn instrument_mcount(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.instrument_mcount, v)
    }
    pub(super) fn instrument_xray(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_instrument_xray(&mut cg.instrument_xray, v)
    }
    pub(super) fn internal_testing_features(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.internal_testing_features, v)
    }
    pub(super) fn large_data_threshold(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.large_data_threshold, v)
    }
    pub(super) fn layout_seed(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.layout_seed, v)
    }
    pub(super) fn link_directives(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.link_directives, v)
    }
    pub(super) fn link_native_libraries(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.link_native_libraries, v)
    }
    pub(super) fn link_only(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.link_only, v)
    }
    pub(super) fn lint_llvm_ir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.lint_llvm_ir, v)
    }
    pub(super) fn lint_mir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.lint_mir, v)
    }
    pub(super) fn lint_rust_version(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_rust_version(&mut cg.lint_rust_version, v)
    }
    pub(super) fn llvm_module_flag(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_llvm_module_flag(&mut cg.llvm_module_flag, v)
    }
    pub(super) fn llvm_plugins(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.llvm_plugins, v)
    }
    pub(super) fn llvm_time_trace(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.llvm_time_trace, v)
    }
    pub(super) fn llvm_writable(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.llvm_writable, v)
    }
    pub(super) fn location_detail(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_location_detail(&mut cg.location_detail, v)
    }
    pub(super) fn ls(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.ls, v)
    }
    pub(super) fn macro_backtrace(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.macro_backtrace, v)
    }
    pub(super) fn macro_stats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.macro_stats, v)
    }
    pub(super) fn maximal_hir_to_mir_coverage(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.maximal_hir_to_mir_coverage, v)
    }
    pub(super) fn merge_functions(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_merge_functions(&mut cg.merge_functions, v)
    }
    pub(super) fn meta_stats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.meta_stats, v)
    }
    pub(super) fn metrics_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.metrics_dir, v)
    }
    pub(super) fn min_function_alignment(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_align(&mut cg.min_function_alignment, v)
    }
    pub(super) fn min_recursion_limit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.min_recursion_limit, v)
    }
    pub(super) fn mir_enable_passes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list_with_polarity(&mut cg.mir_enable_passes, v)
    }
    pub(super) fn mir_include_spans(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_mir_include_spans(&mut cg.mir_include_spans, v)
    }
    pub(super) fn mir_opt_bisect_limit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.mir_opt_bisect_limit, v)
    }
    pub(super) fn mir_opt_level(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.mir_opt_level, v)
    }
    pub(super) fn mir_preserve_ub(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.mir_preserve_ub, v)
    }
    pub(super) fn mir_strip_debuginfo(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_mir_strip_debuginfo(&mut cg.mir_strip_debuginfo,
            v)
    }
    pub(super) fn move_size_limit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.move_size_limit, v)
    }
    pub(super) fn mutable_noalias(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.mutable_noalias, v)
    }
    pub(super) fn namespaced_crates(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.namespaced_crates, v)
    }
    pub(super) fn next_solver(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_next_solver_config(&mut cg.next_solver, v)
    }
    pub(super) fn nll_facts(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.nll_facts, v)
    }
    pub(super) fn nll_facts_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.nll_facts_dir, v)
    }
    pub(super) fn no_analysis(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_analysis, v)
    }
    pub(super) fn no_codegen(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_codegen, v)
    }
    pub(super) fn no_generate_arange_section(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_generate_arange_section, v)
    }
    pub(super) fn no_implied_bounds_compat(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.no_implied_bounds_compat, v)
    }
    pub(super) fn no_leak_check(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_leak_check, v)
    }
    pub(super) fn no_link(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_link, v)
    }
    pub(super) fn no_parallel_backend(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_parallel_backend, v)
    }
    pub(super) fn no_profiler_runtime(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_profiler_runtime, v)
    }
    pub(super) fn no_steal_thir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.no_steal_thir, v)
    }
    pub(super) fn no_trait_vptr(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.no_trait_vptr, v)
    }
    pub(super) fn no_unique_section_names(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.no_unique_section_names, v)
    }
    pub(super) fn normalize_docs(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.normalize_docs, v)
    }
    pub(super) fn offload(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_offload(&mut cg.offload, v)
    }
    pub(super) fn on_broken_pipe(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_on_broken_pipe(&mut cg.on_broken_pipe, v)
    }
    pub(super) fn osx_rpath_install_name(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.osx_rpath_install_name, v)
    }
    pub(super) fn packed_bundled_libs(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.packed_bundled_libs, v)
    }
    pub(super) fn packed_stack(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.packed_stack, v)
    }
    pub(super) fn panic_abort_tests(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.panic_abort_tests, v)
    }
    pub(super) fn panic_in_drop(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_panic_strategy(&mut cg.panic_in_drop, v)
    }
    pub(super) fn parse_crate_root_only(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.parse_crate_root_only, v)
    }
    pub(super) fn patchable_function_entry(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_patchable_function_entry(&mut cg.patchable_function_entry,
            v)
    }
    pub(super) fn plt(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.plt, v)
    }
    pub(super) fn polonius(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_polonius(&mut cg.polonius, v)
    }
    pub(super) fn pre_link_arg(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string_push(&mut cg.pre_link_args, v)
    }
    pub(super) fn pre_link_args(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_list(&mut cg.pre_link_args, v)
    }
    pub(super) fn precise_enum_drop_elaboration(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.precise_enum_drop_elaboration, v)
    }
    pub(super) fn print_codegen_stats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.print_codegen_stats, v)
    }
    pub(super) fn print_codegen_stats_json(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.print_codegen_stats_json, v)
    }
    pub(super) fn print_llvm_passes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.print_llvm_passes, v)
    }
    pub(super) fn print_mono_items(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.print_mono_items, v)
    }
    pub(super) fn print_type_sizes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.print_type_sizes, v)
    }
    pub(super) fn proc_macro_backtrace(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.proc_macro_backtrace, v)
    }
    pub(super) fn proc_macro_execution_strategy(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_proc_macro_execution_strategy(&mut cg.proc_macro_execution_strategy,
            v)
    }
    pub(super) fn profile_closures(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.profile_closures, v)
    }
    pub(super) fn profile_sample_use(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.profile_sample_use, v)
    }
    pub(super) fn profiler_runtime(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.profiler_runtime, v)
    }
    pub(super) fn query_dep_graph(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.query_dep_graph, v)
    }
    pub(super) fn randomize_layout(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.randomize_layout, v)
    }
    pub(super) fn reg_struct_return(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.reg_struct_return, v)
    }
    pub(super) fn regparm(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.regparm, v)
    }
    pub(super) fn relax_elf_relocations(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.relax_elf_relocations, v)
    }
    pub(super) fn remap_cwd_prefix(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.remap_cwd_prefix, v)
    }
    pub(super) fn remark_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.remark_dir, v)
    }
    pub(super) fn retpoline(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.retpoline, v)
    }
    pub(super) fn retpoline_external_thunk(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.retpoline_external_thunk, v)
    }
    pub(super) fn sanitizer(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_sanitizers(&mut cg.sanitizer, v)
    }
    pub(super) fn sanitizer_cfi_canonical_jump_tables(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.sanitizer_cfi_canonical_jump_tables,
            v)
    }
    pub(super) fn sanitizer_cfi_generalize_pointers(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.sanitizer_cfi_generalize_pointers,
            v)
    }
    pub(super) fn sanitizer_cfi_normalize_integers(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.sanitizer_cfi_normalize_integers,
            v)
    }
    pub(super) fn sanitizer_dataflow_abilist(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_comma_list(&mut cg.sanitizer_dataflow_abilist, v)
    }
    pub(super) fn sanitizer_kcfi_arity(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.sanitizer_kcfi_arity, v)
    }
    pub(super) fn sanitizer_memory_track_origins(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_sanitizer_memory_track_origins(&mut cg.sanitizer_memory_track_origins,
            v)
    }
    pub(super) fn sanitizer_recover(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_sanitizers(&mut cg.sanitizer_recover, v)
    }
    pub(super) fn saturating_float_casts(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.saturating_float_casts, v)
    }
    pub(super) fn self_profile(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_switch_with_opt_path(&mut cg.self_profile, v)
    }
    pub(super) fn self_profile_counter(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_string(&mut cg.self_profile_counter, v)
    }
    pub(super) fn self_profile_events(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_comma_list(&mut cg.self_profile_events, v)
    }
    pub(super) fn share_generics(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.share_generics, v)
    }
    pub(super) fn shell_argfiles(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.shell_argfiles, v)
    }
    pub(super) fn simulate_remapped_rust_src_base(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.simulate_remapped_rust_src_base,
            v)
    }
    pub(super) fn small_data_threshold(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_number(&mut cg.small_data_threshold, v)
    }
    pub(super) fn span_debug(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.span_debug, v)
    }
    pub(super) fn span_free_formats(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.span_free_formats, v)
    }
    pub(super) fn split_dwarf_inlining(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.split_dwarf_inlining, v)
    }
    pub(super) fn split_dwarf_kind(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_split_dwarf_kind(&mut cg.split_dwarf_kind, v)
    }
    pub(super) fn split_dwarf_out_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_pathbuf(&mut cg.split_dwarf_out_dir, v)
    }
    pub(super) fn split_lto_unit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.split_lto_unit, v)
    }
    pub(super) fn src_hash_algorithm(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_src_file_hash(&mut cg.src_hash_algorithm, v)
    }
    pub(super) fn stack_protector(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_stack_protector(&mut cg.stack_protector, v)
    }
    pub(super) fn staticlib_allow_rdylib_deps(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.staticlib_allow_rdylib_deps, v)
    }
    pub(super) fn staticlib_hide_internal_symbols(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.staticlib_hide_internal_symbols, v)
    }
    pub(super) fn staticlib_rename_internal_symbols(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.staticlib_rename_internal_symbols, v)
    }
    pub(super) fn staticlib_prefer_dynamic(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.staticlib_prefer_dynamic, v)
    }
    pub(super) fn strict_init_checks(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.strict_init_checks, v)
    }
    pub(super) fn teach(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.teach, v)
    }
    pub(super) fn temps_dir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.temps_dir, v)
    }
    pub(super) fn terminal_urls(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_terminal_url(&mut cg.terminal_urls, v)
    }
    pub(super) fn thinlto(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.thinlto, v)
    }
    pub(super) fn threads(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_threads(&mut cg.threads, v)
    }
    pub(super) fn time_llvm_passes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.time_llvm_passes, v)
    }
    pub(super) fn time_passes(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.time_passes, v)
    }
    pub(super) fn time_passes_format(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_time_passes_format(&mut cg.time_passes_format, v)
    }
    pub(super) fn tiny_const_eval_limit(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.tiny_const_eval_limit, v)
    }
    pub(super) fn tls_model(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_tls_model(&mut cg.tls_model, v)
    }
    pub(super) fn trace_macros(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.trace_macros, v)
    }
    pub(super) fn track_diagnostics(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.track_diagnostics, v)
    }
    pub(super) fn translate_remapped_path_to_local_path(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.translate_remapped_path_to_local_path,
            v)
    }
    pub(super) fn trap_unreachable(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.trap_unreachable, v)
    }
    pub(super) fn treat_err_as_bug(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_treat_err_as_bug(&mut cg.treat_err_as_bug, v)
    }
    pub(super) fn trim_diagnostic_paths(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.trim_diagnostic_paths, v)
    }
    pub(super) fn tune_cpu(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_string(&mut cg.tune_cpu, v)
    }
    pub(super) fn typing_mode_post_typeck_until_borrowck(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.typing_mode_post_typeck_until_borrowck,
            v)
    }
    pub(super) fn ub_checks(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.ub_checks, v)
    }
    pub(super) fn ui_testing(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.ui_testing, v)
    }
    pub(super) fn uninit_const_chunk_threshold(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_number(&mut cg.uninit_const_chunk_threshold, v)
    }
    pub(super) fn unleash_the_miri_inside_of_you(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.unleash_the_miri_inside_of_you, v)
    }
    pub(super) fn unpretty(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_unpretty(&mut cg.unpretty, v)
    }
    pub(super) fn unsound_mir_opts(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.unsound_mir_opts, v)
    }
    pub(super) fn unstable_options(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_no_value(&mut cg.unstable_options, v)
    }
    pub(super) fn use_ctors_section(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.use_ctors_section, v)
    }
    pub(super) fn use_sync_unwind(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_opt_bool(&mut cg.use_sync_unwind, v)
    }
    pub(super) fn validate_mir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.validate_mir, v)
    }
    pub(super) fn verbose_asm(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.verbose_asm, v)
    }
    pub(super) fn verbose_internals(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.verbose_internals, v)
    }
    pub(super) fn verify_llvm_ir(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.verify_llvm_ir, v)
    }
    pub(super) fn virtual_function_elimination(cg:
            &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.virtual_function_elimination, v)
    }
    pub(super) fn wasi_exec_model(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_wasi_exec_model(&mut cg.wasi_exec_model, v)
    }
    pub(super) fn wasm_c_abi(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_wasm_c_abi(&mut cg.wasm_c_abi, v)
    }
    pub(super) fn write_long_types_to_disk(cg: &mut super::UnstableOptions,
        _collected: &mut super::CollectedOptions, v: Option<&str>,
        _index: usize) -> bool {
        super::parse::parse_bool(&mut cg.write_long_types_to_disk, v)
    }
}options! {
2222    UnstableOptions, UnstableOptionsTargetModifiers, Z_OPTIONS, dbopts, "Z", "unstable",
2223
2224    // If you add a new option, please update:
2225    // - compiler/rustc_interface/src/tests.rs
2226    // - src/doc/unstable-book/src/compiler-flags
2227
2228    // tidy-alphabetical-start
2229    allow_features: Option<Vec<String>> = (None, parse_opt_comma_list, [TRACKED],
2230        "only allow the listed language features to be enabled in code (comma separated)"),
2231    // the real parser is at the `setter_for` macro, to allow `-Z` and `-C` options to
2232    // work together.
2233    allow_partial_mitigations: () = ((), parse_allow_partial_mitigations, [UNTRACKED],
2234        "Allow mitigations not enabled for all dependency crates (comma separated list)"),
2235    always_encode_mir: bool = (false, parse_bool, [TRACKED],
2236        "encode MIR of all functions into the crate metadata (default: no)"),
2237    annotate_moves: AnnotateMoves = (AnnotateMoves::Disabled, parse_annotate_moves, [TRACKED],
2238        "emit debug info for compiler-generated move and copy operations \
2239        to make them visible in profilers. Can be a boolean or a size limit in bytes (default: disabled)"),
2240    assert_incr_state: Option<IncrementalStateAssertion> = (None, parse_assert_incr_state, [UNTRACKED],
2241        "assert that the incremental cache is in given state: \
2242         either `loaded` or `not-loaded`."),
2243    assume_incomplete_release: bool = (false, parse_bool, [TRACKED],
2244        "make cfg(version) treat the current version as incomplete (default: no)"),
2245    assumptions_on_binders: bool = (false, parse_bool, [TRACKED],
2246        "allow deducing higher-ranked outlives assumptions from all binders (`for<'a>`)"),
2247    autodiff: Vec<crate::config::AutoDiff> = (Vec::new(), parse_autodiff, [TRACKED],
2248        "a list of autodiff flags to enable
2249        Mandatory setting:
2250        `=Enable`
2251        Optional extra settings:
2252        `=PrintTA`
2253        `=PrintAA`
2254        `=PrintPerf`
2255        `=PrintSteps`
2256        `=PrintModBefore`
2257        `=PrintModAfter`
2258        `=PrintModFinal`
2259        `=PrintPasses`,
2260        `=NoPostopt`
2261        `=LooseTypes`
2262        `=Inline`
2263        Multiple options can be combined with commas."),
2264    #[rustc_lint_opt_deny_field_access("use `Session::binary_dep_depinfo` instead of this field")]
2265    binary_dep_depinfo: bool = (false, parse_bool, [TRACKED],
2266        "include artifacts (sysroot, crate dependencies) used during compilation in dep-info \
2267        (default: no)"),
2268    box_noalias: bool = (true, parse_bool, [TRACKED],
2269        "emit noalias metadata for box (default: yes)"),
2270    branch_protection: Option<BranchProtection> = (None, parse_branch_protection, [TRACKED] { TARGET_MODIFIER: BranchProtection },
2271        "set options for branch target identification and pointer authentication on AArch64"),
2272    build_sdylib_interface: bool = (false, parse_bool, [UNTRACKED],
2273        "whether the stable interface is being built"),
2274    cache_proc_macros: bool = (false, parse_bool, [TRACKED],
2275        "cache the results of derive proc macro invocations (potentially unsound!) (default: no"),
2276    cf_protection: CFProtection = (CFProtection::None, parse_cfprotection, [TRACKED],
2277        "instrument control-flow architecture protection"),
2278    check_cfg_all_expected: bool = (false, parse_bool, [UNTRACKED],
2279        "show all expected values in check-cfg diagnostics (default: no)"),
2280    checksum_hash_algorithm: Option<SourceFileHashAlgorithm> = (None, parse_cargo_src_file_hash, [TRACKED],
2281        "hash algorithm of source files used to check freshness in cargo (`blake3` or `sha256`)"),
2282    codegen_backend: Option<String> = (None, parse_opt_string, [TRACKED],
2283        "the backend to use"),
2284    codegen_emit_retag: Option<CodegenRetagOptions> = (None, parse_codegen_retag_options, [TRACKED],
2285        "emit retag function calls in generated code"),
2286    codegen_source_order: bool = (false, parse_bool, [UNTRACKED],
2287        "emit mono items in the order of spans in source files (default: no)"),
2288    contract_checks: Option<bool> = (None, parse_opt_bool, [TRACKED],
2289        "emit runtime checks for contract pre- and post-conditions (default: no)"),
2290    coverage_options: CoverageOptions = (CoverageOptions::default(), parse_coverage_options, [TRACKED],
2291        "control details of coverage instrumentation"),
2292    crate_attr: Vec<String> = (Vec::new(), parse_string_push, [TRACKED],
2293        "inject the given attribute in the crate"),
2294    cross_crate_inline_threshold: InliningThreshold = (InliningThreshold::Sometimes(100), parse_inlining_threshold, [TRACKED],
2295        "threshold to allow cross crate inlining of functions"),
2296    debug_info_type_line_numbers: bool = (false, parse_bool, [TRACKED],
2297        "emit type and line information for additional data types (default: no)"),
2298    debuginfo_compression: DebugInfoCompression = (DebugInfoCompression::None, parse_debuginfo_compression, [TRACKED],
2299        "compress debug info sections (none, zlib, zstd, default: none)"),
2300    debuginfo_for_profiling: bool = (false, parse_bool, [TRACKED],
2301        "emit discriminators and other data necessary for AutoFDO"),
2302    deduplicate_diagnostics: bool = (true, parse_bool, [UNTRACKED],
2303        "deduplicate identical diagnostics (default: yes)"),
2304    default_visibility: Option<SymbolVisibility> = (None, parse_opt_symbol_visibility, [TRACKED],
2305        "overrides the `default_visibility` setting of the target"),
2306    deny_partial_mitigations: () = ((), parse_deny_partial_mitigations, [UNTRACKED],
2307        "Deny mitigations not enabled for all dependency crates (comma separated list)"),
2308    dep_info_omit_d_target: bool = (false, parse_bool, [TRACKED],
2309        "in dep-info output, omit targets for tracking dependencies of the dep-info files \
2310        themselves (default: no)"),
2311    direct_access_external_data: Option<bool> = (None, parse_opt_bool, [TRACKED],
2312        "Direct or use GOT indirect to reference external data symbols"),
2313    disable_fast_paths: bool = (false, parse_bool, [TRACKED],
2314        "disable various performance optimizations in trait solving"),
2315    disable_incr_comp_backend_caching: bool = (false, parse_bool, [TRACKED],
2316        "disable caching of compiled objects by the codegen backend during incremental compilation"),
2317    dual_proc_macros: bool = (false, parse_bool, [TRACKED],
2318        "load proc macros for both target and host, but only link to the target (default: no)"),
2319    dump_dep_graph: bool = (false, parse_bool, [UNTRACKED],
2320        "dump the dependency graph to $RUST_DEP_GRAPH (default: /tmp/dep_graph.gv) \
2321        (default: no)"),
2322    dump_mir: Option<String> = (None, parse_opt_string, [UNTRACKED],
2323        "dump MIR state to file.
2324        `val` is used to select which passes and functions to dump. For example:
2325        `all` matches all passes and functions,
2326        `foo` matches all passes for functions whose name contains 'foo',
2327        `foo & ConstProp` only the 'ConstProp' pass for function names containing 'foo',
2328        `foo | bar` all passes for function names containing 'foo' or 'bar'."),
2329    dump_mir_dataflow: bool = (false, parse_bool, [UNTRACKED],
2330        "in addition to `.mir` files, create graphviz `.dot` files with dataflow results \
2331        (default: no)"),
2332    dump_mir_dir: String = ("mir_dump".to_string(), parse_string, [UNTRACKED],
2333        "the directory the MIR is dumped into (default: `mir_dump`)"),
2334    dump_mir_exclude_alloc_bytes: bool = (false, parse_bool, [UNTRACKED],
2335        "exclude the raw bytes of allocations when dumping MIR (used in tests) (default: no)"),
2336    dump_mir_exclude_pass_number: bool = (false, parse_bool, [UNTRACKED],
2337        "exclude the pass number when dumping MIR (used in tests) (default: no)"),
2338    dump_mir_graphviz: bool = (false, parse_bool, [UNTRACKED],
2339        "in addition to `.mir` files, create graphviz `.dot` files (default: no)"),
2340    dump_mono_stats: SwitchWithOptPath = (SwitchWithOptPath::Disabled,
2341        parse_switch_with_opt_path, [UNTRACKED],
2342        "output statistics about monomorphization collection"),
2343    dump_mono_stats_format: DumpMonoStatsFormat = (DumpMonoStatsFormat::Markdown, parse_dump_mono_stats, [UNTRACKED],
2344        "the format to use for -Z dump-mono-stats (`markdown` (default) or `json`)"),
2345    #[rustc_lint_opt_deny_field_access("use `Session::dwarf_version` instead of this field")]
2346    dwarf_version: Option<u32> = (None, parse_opt_number, [TRACKED],
2347        "version of DWARF debug information to emit (default: 2 or 4, depending on platform)"),
2348    dylib_lto: bool = (false, parse_bool, [UNTRACKED],
2349        "enables LTO for dylib crate type"),
2350    eagerly_emit_delayed_bugs: bool = (false, parse_bool, [UNTRACKED],
2351        "emit delayed bugs eagerly as errors instead of stashing them and emitting \
2352        them only if an error has not been emitted"),
2353    ehcont_guard: bool = (false, parse_bool, [TRACKED],
2354        "generate Windows EHCont Guard tables"),
2355    embed_metadata: bool = (true, parse_bool, [TRACKED],
2356        "embed metadata in rlibs and dylibs (default: yes)"),
2357    embed_source: bool = (false, parse_bool, [TRACKED],
2358        "embed source text in DWARF debug sections (default: no)"),
2359    emit_stack_sizes: bool = (false, parse_bool, [UNTRACKED],
2360        "emit a section containing stack size metadata (default: no)"),
2361    enforce_type_length_limit: bool = (false, parse_bool, [TRACKED],
2362        "enforce the type length limit when monomorphizing instances in codegen"),
2363    experimental_default_bounds: bool = (false, parse_bool, [TRACKED],
2364        "enable default bounds for experimental group of auto traits"),
2365    export_executable_symbols: bool = (false, parse_bool, [TRACKED],
2366        "export symbols from executables, as if they were dynamic libraries"),
2367    external_clangrt: bool = (false, parse_bool, [UNTRACKED],
2368        "rely on user specified linker commands to find clangrt"),
2369    extra_const_ub_checks: bool = (false, parse_bool, [TRACKED],
2370        "turns on more checks to detect const UB, which can be slow (default: no)"),
2371    #[rustc_lint_opt_deny_field_access("use `Session::fewer_names` instead of this field")]
2372    fewer_names: Option<bool> = (None, parse_opt_bool, [TRACKED],
2373        "reduce memory use by retaining fewer names within compilation artifacts (LLVM-IR) \
2374        (default: no)"),
2375    fixed_x18: bool = (false, parse_bool, [TRACKED] { TARGET_MODIFIER: FixedX18 },
2376        "make the x18 register reserved on AArch64 (default: no)"),
2377    flatten_format_args: bool = (true, parse_bool, [TRACKED],
2378        "flatten nested format_args!() and literals into a simplified format_args!() call \
2379        (default: yes)"),
2380    fmt_debug: FmtDebug = (FmtDebug::Full, parse_fmt_debug, [TRACKED],
2381        "how detailed `#[derive(Debug)]` should be. `full` prints types recursively, \
2382        `shallow` prints only type names, `none` prints nothing and disables `{:?}`. (default: `full`)"),
2383    force_unstable_if_unmarked: bool = (false, parse_bool, [TRACKED],
2384        "force all crates to be `rustc_private` unstable (default: no)"),
2385    function_return: FunctionReturn = (FunctionReturn::default(), parse_function_return, [TRACKED],
2386        "replace returns with jumps to `__x86_return_thunk` (default: `keep`)"),
2387    function_sections: Option<bool> = (None, parse_opt_bool, [TRACKED],
2388        "whether each function should go in its own section"),
2389    future_incompat_test: bool = (false, parse_bool, [UNTRACKED],
2390        "forces all lints to be future incompatible, used for internal testing (default: no)"),
2391    graphviz_dark_mode: bool = (false, parse_bool, [UNTRACKED],
2392        "use dark-themed colors in graphviz output (default: no)"),
2393    graphviz_font: String = ("Courier, monospace".to_string(), parse_string, [UNTRACKED],
2394        "use the given `fontname` in graphviz output; can be overridden by setting \
2395        environment variable `RUSTC_GRAPHVIZ_FONT` (default: `Courier, monospace`)"),
2396    has_thread_local: Option<bool> = (None, parse_opt_bool, [TRACKED],
2397        "explicitly enable the `cfg(target_thread_local)` directive"),
2398    help: bool = (false, parse_no_value, [UNTRACKED], "Print unstable compiler options"),
2399    higher_ranked_assumptions: bool = (false, parse_bool, [TRACKED],
2400        "allow deducing higher-ranked outlives assumptions from coroutines when proving auto traits"),
2401    hint_mostly_unused: bool = (false, parse_bool, [TRACKED],
2402        "hint that most of this crate will go unused, to minimize work for uncalled functions"),
2403    human_readable_cgu_names: bool = (false, parse_bool, [TRACKED],
2404        "generate human-readable, predictable names for codegen units (default: no)"),
2405    identify_regions: bool = (false, parse_bool, [UNTRACKED],
2406        "display unnamed regions as `'<id>`, using a non-ident unique id (default: no)"),
2407    ignore_directory_in_diagnostics_source_blocks: Vec<String> = (Vec::new(), parse_string_push, [UNTRACKED],
2408        "do not display the source code block in diagnostics for files in the directory"),
2409    incremental_ignore_spans: bool = (false, parse_bool, [TRACKED],
2410        "ignore spans during ICH computation -- used for testing (default: no)"),
2411    incremental_info: bool = (false, parse_bool, [UNTRACKED],
2412        "print high-level information about incremental reuse (or the lack thereof) \
2413        (default: no)"),
2414    incremental_verify_ich: bool = (false, parse_bool, [UNTRACKED],
2415        "verify extended properties for incr. comp. (default: no):
2416        - hashes of green query instances
2417        - hash collisions of query keys
2418        - hash collisions when creating dep-nodes"),
2419    indirect_branch_cs_prefix: bool = (false, parse_bool, [TRACKED] { TARGET_MODIFIER: IndirectBranchCsPrefix },
2420        "add `cs` prefix to `call` and `jmp` to indirect thunks (default: no)"),
2421    inline_llvm: bool = (true, parse_bool, [TRACKED],
2422        "enable LLVM inlining (default: yes)"),
2423    inline_mir: Option<bool> = (None, parse_opt_bool, [TRACKED],
2424        "enable MIR inlining (default: no)"),
2425    inline_mir_forwarder_threshold: Option<usize> = (None, parse_opt_number, [TRACKED],
2426        "inlining threshold when the caller is a simple forwarding function (default: 30)"),
2427    inline_mir_hint_threshold: Option<usize> = (None, parse_opt_number, [TRACKED],
2428        "inlining threshold for functions with inline hint (default: 100)"),
2429    inline_mir_preserve_debug: Option<bool> = (None, parse_opt_bool, [TRACKED],
2430        "when MIR inlining, whether to preserve debug info for callee variables \
2431        (default: preserve for debuginfo != None, otherwise remove)"),
2432    inline_mir_threshold: Option<usize> = (None, parse_opt_number, [TRACKED],
2433        "a default MIR inlining threshold (default: 50)"),
2434    input_stats: bool = (false, parse_bool, [UNTRACKED],
2435        "print some statistics about AST and HIR (default: no)"),
2436    instrument_mcount: bool = (false, parse_bool, [TRACKED],
2437        "insert function instrument code for mcount-based tracing (default: no)"),
2438    instrument_xray: Option<InstrumentXRay> = (None, parse_instrument_xray, [TRACKED],
2439        "insert function instrument code for XRay-based tracing (default: no)
2440         Optional extra settings:
2441         `=always`
2442         `=never`
2443         `=ignore-loops`
2444         `=instruction-threshold=N`
2445         `=skip-entry`
2446         `=skip-exit`
2447         Multiple options can be combined with commas."),
2448    // The only purpose of this flag is to act as a deterrent:
2449    // Marking a feature that's only meant for testing as internal might not preclude somebody from
2450    // trying to use it in `core` or `std` as both enable various internal features and utilize them
2451    // throughout. This is intentionally a flag not another internal feature as having to modify the
2452    // build cfg in `bootstrap` is arguably scarier than just needing to add `#![feature]`. It also
2453    // stands out a lot more during code review making it easier to get caught.
2454    internal_testing_features: bool = (false, parse_bool, [TRACKED],
2455        "allow certain internal language features to be enabled that help exercise & test the compiler"),
2456    large_data_threshold: Option<u64> = (None, parse_opt_number, [TRACKED],
2457        "set the threshold for objects to be stored in a \"large data\" section \
2458         (only effective with -Ccode-model=medium, default: 65536)"),
2459    layout_seed: Option<u64> = (None, parse_opt_number, [TRACKED],
2460        "seed layout randomization"),
2461    link_directives: bool = (true, parse_bool, [TRACKED],
2462        "honor #[link] directives in the compiled crate (default: yes)"),
2463    link_native_libraries: bool = (true, parse_bool, [UNTRACKED],
2464        "link native libraries in the linker invocation (default: yes)"),
2465    link_only: bool = (false, parse_bool, [TRACKED],
2466        "link the `.rlink` file generated by `-Z no-link` (default: no)"),
2467    lint_llvm_ir: bool = (false, parse_bool, [TRACKED],
2468        "lint LLVM IR (default: no)"),
2469    lint_mir: bool = (false, parse_bool, [UNTRACKED],
2470        "lint MIR before and after each transformation"),
2471    lint_rust_version: Option<RustcVersion> = (None, parse_rust_version, [TRACKED],
2472        "control the minimum rust version for lints"),
2473    llvm_module_flag: Vec<(String, u32, String)> = (Vec::new(), parse_llvm_module_flag, [TRACKED],
2474        "a list of module flags to pass to LLVM (space separated)"),
2475    llvm_plugins: Vec<String> = (Vec::new(), parse_list, [TRACKED],
2476        "a list LLVM plugins to enable (space separated)"),
2477    llvm_time_trace: bool = (false, parse_bool, [UNTRACKED],
2478        "generate JSON tracing data file from LLVM data (default: no)"),
2479    llvm_writable: bool = (false, parse_bool, [TRACKED],
2480        "emit the LLVM writable attribute for mutable reference arguments (default: no)"),
2481    location_detail: LocationDetail = (LocationDetail::all(), parse_location_detail, [TRACKED],
2482        "what location details should be tracked when using caller_location, either \
2483        `none`, or a comma separated list of location details, for which \
2484        valid options are `file`, `line`, and `column` (default: `file,line,column`)"),
2485    ls: Vec<String> = (Vec::new(), parse_list, [UNTRACKED],
2486        "decode and print various parts of the crate metadata for a library crate \
2487        (space separated)"),
2488    macro_backtrace: bool = (false, parse_bool, [UNTRACKED],
2489        "show macro backtraces (default: no)"),
2490    macro_stats: bool = (false, parse_bool, [UNTRACKED],
2491        "print some statistics about macro expansions (default: no)"),
2492    maximal_hir_to_mir_coverage: bool = (false, parse_bool, [TRACKED],
2493        "save as much information as possible about the correspondence between MIR and HIR \
2494        as source scopes (default: no)"),
2495    merge_functions: Option<MergeFunctions> = (None, parse_merge_functions, [TRACKED],
2496        "control the operation of the MergeFunctions LLVM pass, taking \
2497        the same values as the target option of the same name"),
2498    meta_stats: bool = (false, parse_bool, [UNTRACKED],
2499        "gather metadata statistics (default: no)"),
2500    metrics_dir: Option<PathBuf> = (None, parse_opt_pathbuf, [UNTRACKED],
2501        "the directory metrics emitted by rustc are dumped into (implicitly enables default set of metrics)"),
2502    min_function_alignment: Option<Align> = (None, parse_align, [TRACKED],
2503        "align all functions to at least this many bytes. Must be a power of 2"),
2504    min_recursion_limit: Option<usize> = (None, parse_opt_number, [TRACKED],
2505        "set a minimum recursion limit (final limit = max(this, recursion_limit_from_crate))"),
2506    mir_enable_passes: Vec<(String, bool)> = (Vec::new(), parse_list_with_polarity, [TRACKED],
2507        "use like `-Zmir-enable-passes=+DestinationPropagation,-InstSimplify`. Forces the \
2508        specified passes to be enabled, overriding all other checks. In particular, this will \
2509        enable unsound (known-buggy and hence usually disabled) passes without further warning! \
2510        Passes that are not specified are enabled or disabled by other flags as usual."),
2511    mir_include_spans: MirIncludeSpans = (MirIncludeSpans::default(), parse_mir_include_spans, [UNTRACKED],
2512        "include extra comments in mir pretty printing, like line numbers and statement indices, \
2513         details about types, etc. (boolean for all passes, 'nll' to enable in NLL MIR only, default: 'nll')"),
2514    mir_opt_bisect_limit: Option<usize> = (None, parse_opt_number, [TRACKED],
2515        "limit the number of MIR optimization pass executions (global across all bodies). \
2516        Pass executions after this limit are skipped and reported. (default: no limit)"),
2517    #[rustc_lint_opt_deny_field_access("use `Session::mir_opt_level` instead of this field")]
2518    mir_opt_level: Option<usize> = (None, parse_opt_number, [TRACKED],
2519        "MIR optimization level (0-4; default: 1 in non optimized builds and 2 in optimized builds)"),
2520    mir_preserve_ub: bool = (false, parse_bool, [TRACKED],
2521        "keep place mention statements and reads in trivial SwitchInt terminators, which are interpreted \
2522        e.g., by miri; implies -Zmir-opt-level=0 (default: no)"),
2523    mir_strip_debuginfo: MirStripDebugInfo = (MirStripDebugInfo::None, parse_mir_strip_debuginfo, [TRACKED],
2524        "Whether to remove some of the MIR debug info from methods.  Default: None"),
2525    move_size_limit: Option<usize> = (None, parse_opt_number, [TRACKED],
2526        "the size at which the `large_assignments` lint starts to be emitted"),
2527    mutable_noalias: bool = (true, parse_bool, [TRACKED],
2528        "emit noalias metadata for mutable references (default: yes)"),
2529    namespaced_crates: bool = (false, parse_bool, [TRACKED],
2530        "allow crates to be namespaced by other crates (default: no)"),
2531    next_solver: NextSolverConfig = (NextSolverConfig::default(), parse_next_solver_config, [TRACKED],
2532        "enable and configure the next generation trait solver used by rustc"),
2533    nll_facts: bool = (false, parse_bool, [UNTRACKED],
2534        "dump facts from NLL analysis into side files (default: no)"),
2535    nll_facts_dir: String = ("nll-facts".to_string(), parse_string, [UNTRACKED],
2536        "the directory the NLL facts are dumped into (default: `nll-facts`)"),
2537    no_analysis: bool = (false, parse_no_value, [UNTRACKED],
2538        "parse and expand the source, but run no analysis"),
2539    no_codegen: bool = (false, parse_no_value, [TRACKED_NO_CRATE_HASH],
2540        "run all passes except codegen; no output"),
2541    no_generate_arange_section: bool = (false, parse_no_value, [TRACKED],
2542        "omit DWARF address ranges that give faster lookups"),
2543    no_implied_bounds_compat: bool = (false, parse_bool, [TRACKED],
2544        "disable the compatibility version of the `implied_bounds_ty` query"),
2545    no_leak_check: bool = (false, parse_no_value, [UNTRACKED],
2546        "disable the 'leak check' for subtyping; unsound, but useful for tests"),
2547    no_link: bool = (false, parse_no_value, [TRACKED],
2548        "compile without linking"),
2549    no_parallel_backend: bool = (false, parse_no_value, [UNTRACKED],
2550        "run LLVM in non-parallel mode (while keeping codegen-units and ThinLTO)"),
2551    no_profiler_runtime: bool = (false, parse_no_value, [TRACKED],
2552        "prevent automatic injection of the profiler_builtins crate"),
2553    no_steal_thir: bool = (false, parse_bool, [UNTRACKED],
2554        "don't steal the THIR when we're done with it; useful for rustc drivers (default: no)"),
2555    no_trait_vptr: bool = (false, parse_no_value, [TRACKED],
2556        "disable generation of trait vptr in vtable for upcasting"),
2557    no_unique_section_names: bool = (false, parse_bool, [TRACKED],
2558        "do not use unique names for text and data sections when -Z function-sections is used"),
2559    normalize_docs: bool = (false, parse_bool, [TRACKED],
2560        "normalize associated items in rustdoc when generating documentation"),
2561    offload: Vec<crate::config::Offload> = (Vec::new(), parse_offload, [TRACKED],
2562        "a list of offload flags to enable
2563        Mandatory setting:
2564        `=Enable`
2565        Currently the only option available"),
2566    on_broken_pipe: OnBrokenPipe = (OnBrokenPipe::Default, parse_on_broken_pipe, [TRACKED],
2567        "behavior of std::io::ErrorKind::BrokenPipe (SIGPIPE)"),
2568    osx_rpath_install_name: bool = (false, parse_bool, [TRACKED],
2569        "pass `-install_name @rpath/...` to the macOS linker (default: no)"),
2570    packed_bundled_libs: bool = (false, parse_bool, [TRACKED],
2571        "change rlib format to store native libraries as archives"),
2572    packed_stack: bool = (false, parse_bool, [TRACKED],
2573        "use packed stack frames (s390x only) (default: no)"),
2574    panic_abort_tests: bool = (false, parse_bool, [TRACKED],
2575        "support compiling tests with panic=abort (default: no)"),
2576    panic_in_drop: PanicStrategy = (PanicStrategy::Unwind, parse_panic_strategy, [TRACKED],
2577        "panic strategy for panics in drops"),
2578    parse_crate_root_only: bool = (false, parse_bool, [UNTRACKED],
2579        "parse the crate root file only; do not parse other files, compile, assemble, or link \
2580        (default: no)"),
2581    patchable_function_entry: PatchableFunctionEntry = (PatchableFunctionEntry::default(), parse_patchable_function_entry, [TRACKED],
2582        "nop padding at function entry"),
2583    plt: Option<bool> = (None, parse_opt_bool, [TRACKED],
2584        "whether to use the PLT when calling into shared libraries;
2585        only has effect for PIC code on systems with ELF binaries
2586        (default: PLT is disabled if full relro is enabled on x86_64)"),
2587    polonius: Polonius = (Polonius::default(), parse_polonius, [TRACKED],
2588        "enable polonius-based borrow-checker (default: no)"),
2589    pre_link_arg: (/* redirected to pre_link_args */) = ((), parse_string_push, [UNTRACKED],
2590        "a single extra argument to prepend the linker invocation (can be used several times)"),
2591    pre_link_args: Vec<String> = (Vec::new(), parse_list, [UNTRACKED],
2592        "extra arguments to prepend to the linker invocation (space separated)"),
2593    precise_enum_drop_elaboration: bool = (true, parse_bool, [TRACKED],
2594        "use a more precise version of drop elaboration for matches on enums (default: yes). \
2595        This results in better codegen, but has caused miscompilations on some tier 2 platforms. \
2596        See #77382 and #74551."),
2597    #[rustc_lint_opt_deny_field_access("use `Session::print_codegen_stats` instead of this field")]
2598    print_codegen_stats: bool = (false, parse_bool, [UNTRACKED],
2599        "print codegen statistics (default: no)"),
2600    #[rustc_lint_opt_deny_field_access("use `Session::print_llvm_stats_json` instead of this field")]
2601    print_codegen_stats_json: Option<String> = (None, parse_opt_string, [UNTRACKED],
2602        "print codegen statistics in JSON to a file (default: no)"),
2603    print_llvm_passes: bool = (false, parse_bool, [UNTRACKED],
2604        "print the LLVM optimization passes being run (default: no)"),
2605    print_mono_items: bool = (false, parse_bool, [UNTRACKED],
2606        "print the result of the monomorphization collection pass (default: no)"),
2607    print_type_sizes: bool = (false, parse_bool, [UNTRACKED],
2608        "print layout information for each type encountered (default: no)"),
2609    proc_macro_backtrace: bool = (false, parse_bool, [UNTRACKED],
2610         "show backtraces for panics during proc-macro execution (default: no)"),
2611    proc_macro_execution_strategy: ProcMacroExecutionStrategy = (ProcMacroExecutionStrategy::SameThread,
2612        parse_proc_macro_execution_strategy, [UNTRACKED],
2613        "how to run proc-macro code (default: same-thread)"),
2614    profile_closures: bool = (false, parse_no_value, [UNTRACKED],
2615        "profile size of closures"),
2616    profile_sample_use: Option<PathBuf> = (None, parse_opt_pathbuf, [TRACKED],
2617        "use the given `.prof` file for sampled profile-guided optimization (also known as AutoFDO)"),
2618    profiler_runtime: String = (String::from("profiler_builtins"), parse_string, [TRACKED],
2619        "name of the profiler runtime crate to automatically inject (default: `profiler_builtins`)"),
2620    query_dep_graph: bool = (false, parse_bool, [UNTRACKED],
2621        "enable queries of the dependency graph for regression testing (default: no)"),
2622    randomize_layout: bool = (false, parse_bool, [TRACKED],
2623        "randomize the layout of types (default: no)"),
2624    reg_struct_return: bool = (false, parse_bool, [TRACKED] { TARGET_MODIFIER: RegStructReturn },
2625        "On x86-32 targets, it overrides the default ABI to return small structs in registers.
2626        It is UNSOUND to link together crates that use different values for this flag!"),
2627    regparm: Option<u32> = (None, parse_opt_number, [TRACKED] { TARGET_MODIFIER: Regparm },
2628        "On x86-32 targets, setting this to N causes the compiler to pass N arguments \
2629        in registers EAX, EDX, and ECX instead of on the stack for\
2630        \"C\", \"cdecl\", and \"stdcall\" fn.\
2631        It is UNSOUND to link together crates that use different values for this flag!"),
2632    relax_elf_relocations: Option<bool> = (None, parse_opt_bool, [TRACKED],
2633        "whether ELF relocations can be relaxed"),
2634    remap_cwd_prefix: Option<PathBuf> = (None, parse_opt_pathbuf, [TRACKED],
2635        "remap paths under the current working directory to this path prefix"),
2636    remark_dir: Option<PathBuf> = (None, parse_opt_pathbuf, [UNTRACKED],
2637        "directory into which to write optimization remarks (if not specified, they will be \
2638written to standard error output)"),
2639    retpoline: bool = (false, parse_bool, [TRACKED] { TARGET_MODIFIER: Retpoline },
2640        "enables retpoline-indirect-branches and retpoline-indirect-calls target features (default: no)"),
2641    retpoline_external_thunk: bool = (false, parse_bool, [TRACKED] { TARGET_MODIFIER: RetpolineExternalThunk },
2642        "enables retpoline-external-thunk, retpoline-indirect-branches and retpoline-indirect-calls \
2643        target features (default: no)"),
2644    #[rustc_lint_opt_deny_field_access("use `Session::sanitizers()` instead of this field")]
2645    sanitizer: SanitizerSet = (SanitizerSet::empty(), parse_sanitizers, [TRACKED] { TARGET_MODIFIER: Sanitizer },
2646        "use a sanitizer"),
2647    sanitizer_cfi_canonical_jump_tables: Option<bool> = (Some(true), parse_opt_bool, [TRACKED],
2648        "enable canonical jump tables (default: yes)"),
2649    sanitizer_cfi_generalize_pointers: Option<bool> = (None, parse_opt_bool, [TRACKED],
2650        "enable generalizing pointer types (default: no)"),
2651    sanitizer_cfi_normalize_integers: Option<bool> = (None, parse_opt_bool, [TRACKED] { TARGET_MODIFIER: SanitizerCfiNormalizeIntegers },
2652        "enable normalizing integer types (default: no)"),
2653    sanitizer_dataflow_abilist: Vec<String> = (Vec::new(), parse_comma_list, [TRACKED],
2654        "additional ABI list files that control how shadow parameters are passed (comma separated)"),
2655    sanitizer_kcfi_arity: Option<bool> = (None, parse_opt_bool, [TRACKED],
2656        "enable KCFI arity indicator (default: no)"),
2657    sanitizer_memory_track_origins: usize = (0, parse_sanitizer_memory_track_origins, [TRACKED],
2658        "enable origins tracking in MemorySanitizer"),
2659    sanitizer_recover: SanitizerSet = (SanitizerSet::empty(), parse_sanitizers, [TRACKED],
2660        "enable recovery for selected sanitizers"),
2661    saturating_float_casts: Option<bool> = (None, parse_opt_bool, [TRACKED],
2662        "make float->int casts UB-free: numbers outside the integer type's range are clipped to \
2663        the max/min integer respectively, and NaN is mapped to 0 (default: yes)"),
2664    self_profile: SwitchWithOptPath = (SwitchWithOptPath::Disabled,
2665        parse_switch_with_opt_path, [UNTRACKED],
2666        "run the self profiler and output the raw event data"),
2667    self_profile_counter: String = ("wall-time".to_string(), parse_string, [UNTRACKED],
2668        "counter used by the self profiler (default: `wall-time`), one of:
2669        `wall-time` (monotonic clock, i.e. `std::time::Instant`)
2670        `instructions:u` (retired instructions, userspace-only)
2671        `instructions-minus-irqs:u` (subtracting hardware interrupt counts for extra accuracy)"
2672    ),
2673    /// keep this in sync with the event filter names in librustc_data_structures/profiling.rs
2674    self_profile_events: Option<Vec<String>> = (None, parse_opt_comma_list, [UNTRACKED],
2675        "specify the events recorded by the self profiler;
2676        for example: `-Z self-profile-events=default,query-keys`
2677        all options: none, all, default, generic-activity, query-provider, query-cache-hit
2678                     query-blocked, incr-cache-load, incr-result-hashing, query-keys, function-args, args, llvm, artifact-sizes"),
2679    share_generics: Option<bool> = (None, parse_opt_bool, [TRACKED],
2680        "make the current crate share its generic instantiations"),
2681    shell_argfiles: bool = (false, parse_bool, [UNTRACKED],
2682        "allow argument files to be specified with POSIX \"shell-style\" argument quoting"),
2683    simulate_remapped_rust_src_base: Option<PathBuf> = (None, parse_opt_pathbuf, [TRACKED],
2684        "simulate the effect of remap-debuginfo = true at bootstrapping by remapping path \
2685        to rust's source base directory. only meant for testing purposes"),
2686    small_data_threshold: Option<usize> = (None, parse_opt_number, [TRACKED],
2687        "Set the threshold for objects to be stored in a \"small data\" section"),
2688    span_debug: bool = (false, parse_bool, [UNTRACKED],
2689        "forward proc_macro::Span's `Debug` impl to `Span`"),
2690    /// o/w tests have closure@path
2691    span_free_formats: bool = (false, parse_bool, [UNTRACKED],
2692        "exclude spans when debug-printing compiler state (default: no)"),
2693    split_dwarf_inlining: bool = (false, parse_bool, [TRACKED],
2694        "provide minimal debug info in the object/executable to facilitate online \
2695         symbolication/stack traces in the absence of .dwo/.dwp files when using Split DWARF"),
2696    split_dwarf_kind: SplitDwarfKind = (SplitDwarfKind::Split, parse_split_dwarf_kind, [TRACKED],
2697        "split dwarf variant (only if -Csplit-debuginfo is enabled and on relevant platform)
2698        (default: `split`)
2699
2700        `split`: sections which do not require relocation are written into a DWARF object (`.dwo`)
2701                 file which is ignored by the linker
2702        `single`: sections which do not require relocation are written into object file but ignored
2703                  by the linker"),
2704    split_dwarf_out_dir : Option<PathBuf> = (None, parse_opt_pathbuf, [TRACKED],
2705        "location for writing split DWARF objects (`.dwo`) if enabled"),
2706    split_lto_unit: Option<bool> = (None, parse_opt_bool, [TRACKED],
2707        "enable LTO unit splitting (default: no)"),
2708    src_hash_algorithm: Option<SourceFileHashAlgorithm> = (None, parse_src_file_hash, [TRACKED],
2709        "hash algorithm of source files in debug info (`md5`, `sha1`, or `sha256`)"),
2710    #[rustc_lint_opt_deny_field_access("use `Session::stack_protector` instead of this field")]
2711    stack_protector: StackProtector = (StackProtector::None, parse_stack_protector, [TRACKED] { MITIGATION: StackProtector },
2712        "control stack smash protection strategy (`rustc --print stack-protector-strategies` for details)"),
2713    staticlib_allow_rdylib_deps: bool = (false, parse_bool, [TRACKED],
2714        "allow staticlibs to have rust dylib dependencies"),
2715    staticlib_hide_internal_symbols: bool = (false, parse_bool, [TRACKED],
2716        "hide non-exported Rust symbols when building staticlibs by setting STV_HIDDEN"),
2717    staticlib_rename_internal_symbols: bool = (false, parse_bool, [TRACKED],
2718        "rename non-exported Rust symbols when building staticlibs to avoid conflicts"),
2719    staticlib_prefer_dynamic: bool = (false, parse_bool, [TRACKED],
2720        "prefer dynamic linking to static linking for staticlibs (default: no)"),
2721    strict_init_checks: bool = (false, parse_bool, [TRACKED],
2722        "control if mem::uninitialized and mem::zeroed panic on more UB"),
2723    #[rustc_lint_opt_deny_field_access("use `Session::teach` instead of this field")]
2724    teach: bool = (false, parse_bool, [TRACKED],
2725        "show extended diagnostic help (default: no)"),
2726    temps_dir: Option<String> = (None, parse_opt_string, [UNTRACKED],
2727        "the directory the intermediate files are written to"),
2728    terminal_urls: TerminalUrl = (TerminalUrl::No, parse_terminal_url, [UNTRACKED],
2729        "use the OSC 8 hyperlink terminal specification to print hyperlinks in the compiler output"),
2730    #[rustc_lint_opt_deny_field_access("use `Session::lto` instead of this field")]
2731    thinlto: Option<bool> = (None, parse_opt_bool, [TRACKED],
2732        "enable ThinLTO when possible"),
2733    /// We default to None here since we want to behave like
2734    /// a sequential compiler for now. This'll likely be adjusted
2735    /// in the future. Note that -Zthreads=0 is the way to get
2736    /// the num_cpus behavior.
2737    #[rustc_lint_opt_deny_field_access("use `Session::threads` instead of this field")]
2738    threads: Option<usize> = (None, parse_threads, [UNTRACKED],
2739        "use a thread pool with N threads"),
2740    time_llvm_passes: bool = (false, parse_bool, [UNTRACKED],
2741        "measure time of each LLVM pass (default: no)"),
2742    time_passes: bool = (false, parse_bool, [UNTRACKED],
2743        "measure time of each rustc pass (default: no)"),
2744    time_passes_format: TimePassesFormat = (TimePassesFormat::Text, parse_time_passes_format, [UNTRACKED],
2745        "the format to use for -Z time-passes (`text` (default) or `json`)"),
2746    tiny_const_eval_limit: bool = (false, parse_bool, [TRACKED],
2747        "sets a tiny, non-configurable limit for const eval; useful for compiler tests"),
2748    #[rustc_lint_opt_deny_field_access("use `Session::tls_model` instead of this field")]
2749    tls_model: Option<TlsModel> = (None, parse_tls_model, [TRACKED],
2750        "choose the TLS model to use (`rustc --print tls-models` for details)"),
2751    trace_macros: bool = (false, parse_bool, [UNTRACKED],
2752        "for every macro invocation, print its name and arguments (default: no)"),
2753    track_diagnostics: bool = (false, parse_bool, [UNTRACKED],
2754        "tracks where in rustc a diagnostic was emitted"),
2755    translate_remapped_path_to_local_path: bool = (true, parse_bool, [TRACKED],
2756        "translate remapped paths into local paths when possible (default: yes)"),
2757    trap_unreachable: Option<bool> = (None, parse_opt_bool, [TRACKED],
2758        "generate trap instructions for unreachable intrinsics (default: use target setting, usually yes)"),
2759    treat_err_as_bug: Option<NonZero<usize>> = (None, parse_treat_err_as_bug, [TRACKED],
2760        "treat the `val`th error that occurs as bug (default if not specified: 0 - don't treat errors as bugs. \
2761        default if specified without a value: 1 - treat the first error as bug)"),
2762    trim_diagnostic_paths: bool = (true, parse_bool, [UNTRACKED],
2763        "in diagnostics, use heuristics to shorten paths referring to items"),
2764    tune_cpu: Option<String> = (None, parse_opt_string, [TRACKED],
2765        "select processor to schedule for (`rustc --print target-cpus` for details)"),
2766    #[rustc_lint_opt_deny_field_access("use `TyCtxt::use_typing_mode_post_typeck` instead of this field")]
2767    typing_mode_post_typeck_until_borrowck: bool = (false, parse_bool, [TRACKED],
2768        "enable `TypingMode::PostTypeckUntilBorrowck`, changing the way opaque types are handled during MIR borrowck"),
2769    #[rustc_lint_opt_deny_field_access("use `Session::ub_checks` instead of this field")]
2770    ub_checks: Option<bool> = (None, parse_opt_bool, [TRACKED],
2771        "emit runtime checks for Undefined Behavior (default: -Cdebug-assertions)"),
2772    ui_testing: bool = (false, parse_bool, [UNTRACKED],
2773        "emit compiler diagnostics in a form suitable for UI testing (default: no)"),
2774    uninit_const_chunk_threshold: usize = (16, parse_number, [TRACKED],
2775        "allow generating const initializers with mixed init/uninit chunks, \
2776        and set the maximum number of chunks for which this is allowed (default: 16)"),
2777    unleash_the_miri_inside_of_you: bool = (false, parse_bool, [TRACKED],
2778        "take the brakes off const evaluation. NOTE: this is unsound (default: no)"),
2779    unpretty: Option<String> = (None, parse_unpretty, [UNTRACKED],
2780        "present the input source, unstable (and less-pretty) variants;
2781        `normal`,
2782        `expanded`, `expanded,identified`,
2783        `expanded,hygiene` (with internal representations),
2784        `ast-tree` (raw AST before expansion),
2785        `ast-tree,expanded` (raw AST after expansion),
2786        `hir` (the HIR), `hir,identified`,
2787        `hir,typed` (HIR with types for each node),
2788        `hir-tree` (dump the raw HIR),
2789        `thir-tree`, `thir-flat`,
2790        `mir` (the MIR), or `mir-cfg` (graphviz formatted MIR)"),
2791    unsound_mir_opts: bool = (false, parse_bool, [TRACKED],
2792        "enable unsound and buggy MIR optimizations (default: no)"),
2793    /// This name is kind of confusing: Most unstable options enable something themselves, while
2794    /// this just allows "normal" options to be feature-gated.
2795    ///
2796    /// The main check for `-Zunstable-options` takes place separately from the
2797    /// usual parsing of `-Z` options (see [`crate::config::nightly_options`]),
2798    /// so this boolean value is mostly used for enabling unstable _values_ of
2799    /// stable options. That separate check doesn't handle boolean values, so
2800    /// to avoid an inconsistent state we also forbid them here.
2801    #[rustc_lint_opt_deny_field_access("use `Session::unstable_options` instead of this field")]
2802    unstable_options: bool = (false, parse_no_value, [UNTRACKED],
2803        "adds unstable command line options to rustc interface (default: no)"),
2804    use_ctors_section: Option<bool> = (None, parse_opt_bool, [TRACKED],
2805        "use legacy .ctors section for initializers rather than .init_array"),
2806    use_sync_unwind: Option<bool> = (None, parse_opt_bool, [TRACKED],
2807        "Generate sync unwind tables instead of async unwind tables (default: no)"),
2808    validate_mir: bool = (false, parse_bool, [UNTRACKED],
2809        "validate MIR after each transformation"),
2810    verbose_asm: bool = (false, parse_bool, [TRACKED],
2811        "add descriptive comments from LLVM to the assembly (may change behavior) (default: no)"),
2812    #[rustc_lint_opt_deny_field_access("use `Session::verbose_internals` instead of this field")]
2813    verbose_internals: bool = (false, parse_bool, [TRACKED_NO_CRATE_HASH],
2814        "in general, enable more debug printouts (default: no)"),
2815    #[rustc_lint_opt_deny_field_access("use `Session::verify_llvm_ir` instead of this field")]
2816    verify_llvm_ir: bool = (false, parse_bool, [TRACKED],
2817        "verify LLVM IR (default: no)"),
2818    virtual_function_elimination: bool = (false, parse_bool, [TRACKED],
2819        "enables dead virtual function elimination optimization. \
2820        Requires `-Clto[=[fat,yes]]`"),
2821    wasi_exec_model: Option<WasiExecModel> = (None, parse_wasi_exec_model, [TRACKED],
2822        "whether to build a wasi command or reactor"),
2823    // This option only still exists to provide a more gradual transition path for people who need
2824    // the spec-complaint C ABI to be used.
2825    // FIXME remove this after a couple releases
2826    wasm_c_abi: () = ((), parse_wasm_c_abi, [TRACKED],
2827        "use spec-compliant C ABI for `wasm32-unknown-unknown` (deprecated, always enabled)"),
2828    write_long_types_to_disk: bool = (true, parse_bool, [UNTRACKED],
2829        "whether long type names should be written to files instead of being printed in errors"),
2830    // tidy-alphabetical-end
2831
2832    // If you add a new option, please update:
2833    // - compiler/rustc_interface/src/tests.rs
2834    // - src/doc/unstable-book/src/compiler-flags
2835}