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