Skip to main content

rustc_lint/types/
improper_ctypes.rs

1use std::iter;
2use std::ops::ControlFlow;
3
4use bitflags::bitflags;
5use rustc_abi::VariantIdx;
6use rustc_data_structures::fx::FxHashSet;
7use rustc_errors::{DiagMessage, msg};
8use rustc_hir::def::CtorKind;
9use rustc_hir::intravisit::VisitorExt;
10use rustc_hir::{self as hir, AmbigArg};
11use rustc_middle::bug;
12use rustc_middle::ty::{
13    self, Adt, AdtDef, AdtKind, GenericArgsRef, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable,
14    TypeVisitableExt, Unnormalized,
15};
16use rustc_session::{declare_lint, declare_lint_pass};
17use rustc_span::def_id::LocalDefId;
18use rustc_span::{Span, sym};
19use rustc_target::spec::Os;
20use tracing::debug;
21
22use super::repr_nullable_ptr;
23use crate::lints::{ImproperCTypes, UsesPowerAlignment};
24use crate::{LateContext, LateLintPass, LintContext};
25
26#[doc =
r" The `improper_ctypes` lint detects incorrect use of types in foreign"]
#[doc = r" modules."]
#[doc = r""]
#[doc = r" ### Example"]
#[doc = r""]
#[doc = r" ```rust"]
#[doc = r#" unsafe extern "C" {"#]
#[doc = r"     static STATIC: String;"]
#[doc = r" }"]
#[doc = r" ```"]
#[doc = r""]
#[doc = r" {{produces}}"]
#[doc = r""]
#[doc = r" ### Explanation"]
#[doc = r""]
#[doc =
r" The compiler has several checks to verify that types used in `extern`"]
#[doc = r" blocks are safe and follow certain rules to ensure proper"]
#[doc =
r" compatibility with the foreign interfaces. This lint is issued when it"]
#[doc =
r" detects a probable mistake in a definition. The lint usually should"]
#[doc =
r" provide a description of the issue, along with possibly a hint on how"]
#[doc = r" to resolve it."]
static IMPROPER_CTYPES: &::rustc_lint_defs::Lint =
    &::rustc_lint_defs::Lint {
            name: "IMPROPER_CTYPES",
            default_level: ::rustc_lint_defs::Warn,
            desc: "proper use of libc types in foreign modules",
            is_externally_loaded: false,
            ..::rustc_lint_defs::Lint::default_fields_for_macro()
        };declare_lint! {
27    /// The `improper_ctypes` lint detects incorrect use of types in foreign
28    /// modules.
29    ///
30    /// ### Example
31    ///
32    /// ```rust
33    /// unsafe extern "C" {
34    ///     static STATIC: String;
35    /// }
36    /// ```
37    ///
38    /// {{produces}}
39    ///
40    /// ### Explanation
41    ///
42    /// The compiler has several checks to verify that types used in `extern`
43    /// blocks are safe and follow certain rules to ensure proper
44    /// compatibility with the foreign interfaces. This lint is issued when it
45    /// detects a probable mistake in a definition. The lint usually should
46    /// provide a description of the issue, along with possibly a hint on how
47    /// to resolve it.
48    IMPROPER_CTYPES,
49    Warn,
50    "proper use of libc types in foreign modules"
51}
52
53#[doc = r" The `improper_ctypes_definitions` lint detects incorrect use of"]
#[doc = r" [`extern` function] definitions."]
#[doc = r""]
#[doc =
r" [`extern` function]: https://doc.rust-lang.org/reference/items/functions.html#extern-function-qualifier"]
#[doc = r""]
#[doc = r" ### Example"]
#[doc = r""]
#[doc = r" ```rust"]
#[doc = r" # #![allow(unused)]"]
#[doc = r#" pub extern "C" fn str_type(p: &str) { }"#]
#[doc = r" ```"]
#[doc = r""]
#[doc = r" {{produces}}"]
#[doc = r""]
#[doc = r" ### Explanation"]
#[doc = r""]
#[doc =
r" There are many parameter and return types that may be specified in an"]
#[doc =
r" `extern` function that are not compatible with the given ABI. This"]
#[doc =
r" lint is an alert that these types should not be used. The lint usually"]
#[doc =
r" should provide a description of the issue, along with possibly a hint"]
#[doc = r" on how to resolve it."]
static IMPROPER_CTYPES_DEFINITIONS: &::rustc_lint_defs::Lint =
    &::rustc_lint_defs::Lint {
            name: "IMPROPER_CTYPES_DEFINITIONS",
            default_level: ::rustc_lint_defs::Warn,
            desc: "proper use of libc types in foreign item definitions",
            is_externally_loaded: false,
            ..::rustc_lint_defs::Lint::default_fields_for_macro()
        };declare_lint! {
54    /// The `improper_ctypes_definitions` lint detects incorrect use of
55    /// [`extern` function] definitions.
56    ///
57    /// [`extern` function]: https://doc.rust-lang.org/reference/items/functions.html#extern-function-qualifier
58    ///
59    /// ### Example
60    ///
61    /// ```rust
62    /// # #![allow(unused)]
63    /// pub extern "C" fn str_type(p: &str) { }
64    /// ```
65    ///
66    /// {{produces}}
67    ///
68    /// ### Explanation
69    ///
70    /// There are many parameter and return types that may be specified in an
71    /// `extern` function that are not compatible with the given ABI. This
72    /// lint is an alert that these types should not be used. The lint usually
73    /// should provide a description of the issue, along with possibly a hint
74    /// on how to resolve it.
75    IMPROPER_CTYPES_DEFINITIONS,
76    Warn,
77    "proper use of libc types in foreign item definitions"
78}
79
80#[doc = r" The `uses_power_alignment` lint detects specific `repr(C)`"]
#[doc = r" aggregates on AIX."]
#[doc =
r#" In its platform C ABI, AIX uses the "power" (as in PowerPC) alignment"#]
#[doc =
r" rule (detailed in https://www.ibm.com/docs/en/xl-c-and-cpp-aix/16.1?topic=data-using-alignment-modes#alignment),"]
#[doc = r" which can also be set for XLC by `#pragma align(power)` or"]
#[doc = r" `-qalign=power`. Aggregates with a floating-point type as the"]
#[doc =
r#" recursively first field (as in "at offset 0") modify the layout of"#]
#[doc =
r" *subsequent* fields of the associated structs to use an alignment value"]
#[doc = r" where the floating-point type is aligned on a 4-byte boundary."]
#[doc = r""]
#[doc =
r" Effectively, subsequent floating-point fields act as-if they are `repr(packed(4))`. This"]
#[doc =
r" would be unsound to do in a `repr(C)` type without all the restrictions that come with"]
#[doc =
r" `repr(packed)`. Rust instead chooses a layout that maintains soundness of Rust code, at the"]
#[doc = r" expense of incompatibility with C code."]
#[doc = r""]
#[doc = r" ### Example"]
#[doc = r""]
#[doc = r" ```rust,ignore (fails on non-powerpc64-ibm-aix)"]
#[doc = r" #[repr(C)]"]
#[doc = r" pub struct Floats {"]
#[doc = r"     a: f64,"]
#[doc = r"     b: u8,"]
#[doc = r"     c: f64,"]
#[doc = r" }"]
#[doc = r" ```"]
#[doc = r""]
#[doc = r" This will produce:"]
#[doc = r""]
#[doc = r" ```text"]
#[doc =
r" warning: repr(C) does not follow the power alignment rule. This may affect platform C ABI compatibility for this type"]
#[doc = r"  --> <source>:5:3"]
#[doc = r"   |"]
#[doc = r" 5 |   c: f64,"]
#[doc = r"   |   ^^^^^^"]
#[doc = r"   |"]
#[doc = r"   = note: `#[warn(uses_power_alignment)]` on by default"]
#[doc = r" ```"]
#[doc = r""]
#[doc = r" ### Explanation"]
#[doc = r""]
#[doc = r" The power alignment rule specifies that the above struct has the"]
#[doc = r" following alignment:"]
#[doc = r"  - offset_of!(Floats, a) == 0"]
#[doc = r"  - offset_of!(Floats, b) == 8"]
#[doc = r"  - offset_of!(Floats, c) == 12"]
#[doc = r""]
#[doc =
r" However, Rust currently aligns `c` at `offset_of!(Floats, c) == 16`."]
#[doc =
r" Using offset 12 would be unsound since `f64` generally must be 8-aligned on this target."]
#[doc = r" Thus, a warning is produced for the above struct."]
static USES_POWER_ALIGNMENT: &::rustc_lint_defs::Lint =
    &::rustc_lint_defs::Lint {
            name: "USES_POWER_ALIGNMENT",
            default_level: ::rustc_lint_defs::Warn,
            desc: "Structs do not follow the power alignment rule under repr(C)",
            is_externally_loaded: false,
            ..::rustc_lint_defs::Lint::default_fields_for_macro()
        };declare_lint! {
81    /// The `uses_power_alignment` lint detects specific `repr(C)`
82    /// aggregates on AIX.
83    /// In its platform C ABI, AIX uses the "power" (as in PowerPC) alignment
84    /// rule (detailed in https://www.ibm.com/docs/en/xl-c-and-cpp-aix/16.1?topic=data-using-alignment-modes#alignment),
85    /// which can also be set for XLC by `#pragma align(power)` or
86    /// `-qalign=power`. Aggregates with a floating-point type as the
87    /// recursively first field (as in "at offset 0") modify the layout of
88    /// *subsequent* fields of the associated structs to use an alignment value
89    /// where the floating-point type is aligned on a 4-byte boundary.
90    ///
91    /// Effectively, subsequent floating-point fields act as-if they are `repr(packed(4))`. This
92    /// would be unsound to do in a `repr(C)` type without all the restrictions that come with
93    /// `repr(packed)`. Rust instead chooses a layout that maintains soundness of Rust code, at the
94    /// expense of incompatibility with C code.
95    ///
96    /// ### Example
97    ///
98    /// ```rust,ignore (fails on non-powerpc64-ibm-aix)
99    /// #[repr(C)]
100    /// pub struct Floats {
101    ///     a: f64,
102    ///     b: u8,
103    ///     c: f64,
104    /// }
105    /// ```
106    ///
107    /// This will produce:
108    ///
109    /// ```text
110    /// warning: repr(C) does not follow the power alignment rule. This may affect platform C ABI compatibility for this type
111    ///  --> <source>:5:3
112    ///   |
113    /// 5 |   c: f64,
114    ///   |   ^^^^^^
115    ///   |
116    ///   = note: `#[warn(uses_power_alignment)]` on by default
117    /// ```
118    ///
119    /// ### Explanation
120    ///
121    /// The power alignment rule specifies that the above struct has the
122    /// following alignment:
123    ///  - offset_of!(Floats, a) == 0
124    ///  - offset_of!(Floats, b) == 8
125    ///  - offset_of!(Floats, c) == 12
126    ///
127    /// However, Rust currently aligns `c` at `offset_of!(Floats, c) == 16`.
128    /// Using offset 12 would be unsound since `f64` generally must be 8-aligned on this target.
129    /// Thus, a warning is produced for the above struct.
130    USES_POWER_ALIGNMENT,
131    Warn,
132    "Structs do not follow the power alignment rule under repr(C)"
133}
134
135pub struct ImproperCTypesLint;
#[automatically_derived]
impl ::core::marker::Copy for ImproperCTypesLint { }
#[automatically_derived]
#[doc(hidden)]
unsafe impl ::core::clone::TrivialClone for ImproperCTypesLint { }
#[automatically_derived]
impl ::core::clone::Clone for ImproperCTypesLint {
    #[inline]
    fn clone(&self) -> ImproperCTypesLint { *self }
}
impl ::rustc_lint_defs::LintPass for ImproperCTypesLint {
    fn name(&self) -> &'static str { "ImproperCTypesLint" }
    fn get_lints(&self) -> ::rustc_lint_defs::LintVec {
        ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
                [IMPROPER_CTYPES, IMPROPER_CTYPES_DEFINITIONS,
                        USES_POWER_ALIGNMENT]))
    }
}
impl ImproperCTypesLint {
    #[allow(unused)]
    pub fn lint_vec() -> ::rustc_lint_defs::LintVec {
        ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
                [IMPROPER_CTYPES, IMPROPER_CTYPES_DEFINITIONS,
                        USES_POWER_ALIGNMENT]))
    }
}declare_lint_pass!(ImproperCTypesLint => [
136    IMPROPER_CTYPES,
137    IMPROPER_CTYPES_DEFINITIONS,
138    USES_POWER_ALIGNMENT
139]);
140
141/// A common pattern in this lint is to attempt normalize_erasing_regions,
142/// but keep the original type if it were to fail.
143/// This may or may not be supported in the logic behind the `Unnormalized` wrapper,
144/// (FIXME?)
145/// but it should be enough for non-wrapped types to be as normalised as this lint needs them to be.
146fn maybe_normalize_erasing_regions<'tcx>(
147    cx: &LateContext<'tcx>,
148    value: Unnormalized<'tcx, Ty<'tcx>>,
149) -> Ty<'tcx> {
150    // Use `TypingMode::Borrowck` so the new solver doesn't reveal opaque types since we're now
151    // past hir typeck. If we were to attempt to reveal more opaque types, dropping the
152    // `InferCtxt` would ICE (see #156352).
153    let typing_env = if let Some(body_id) = cx.enclosing_body {
154        let body_def_id = cx.tcx.hir_enclosing_body_owner(body_id.hir_id);
155        ty::TypingEnv::new(cx.param_env, ty::TypingMode::borrowck(cx.tcx, body_def_id))
156    } else {
157        cx.typing_env()
158    };
159    cx.tcx.try_normalize_erasing_regions(typing_env, value).unwrap_or(value.skip_norm_wip())
160}
161
162/// Check a variant of a non-exhaustive enum for improper ctypes
163///
164/// We treat `#[non_exhaustive] enum` as "ensure that code will compile if new variants are added".
165/// This includes linting, on a best-effort basis. There are valid additions that are unlikely.
166///
167/// Adding a data-carrying variant to an existing C-like enum that is passed to C is "unlikely",
168/// so we don't need the lint to account for it.
169/// e.g. going from enum Foo { A, B, C } to enum Foo { A, B, C, D(u32) }.
170pub(crate) fn check_non_exhaustive_variant(
171    non_exhaustive_variant_list: bool,
172    variant: &ty::VariantDef,
173) -> ControlFlow<DiagMessage, ()> {
174    // non_exhaustive suggests it is possible that someone might break ABI
175    // see: https://github.com/rust-lang/rust/issues/44109#issuecomment-537583344
176    // so warn on complex enums being used outside their crate
177    if non_exhaustive_variant_list {
178        // which is why we only warn about really_tagged_union reprs from https://rust.tf/rfc2195
179        // with an enum like `#[repr(u8)] enum Enum { A(DataA), B(DataB), }`
180        // but exempt enums with unit ctors like C's (e.g. from rust-bindgen)
181        if variant_has_complex_ctor(variant) {
182            return ControlFlow::Break(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this enum is non-exhaustive"))msg!("this enum is non-exhaustive"));
183        }
184    }
185
186    if variant.field_list_has_applicable_non_exhaustive() {
187        return ControlFlow::Break(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this enum has non-exhaustive variants"))msg!("this enum has non-exhaustive variants"));
188    }
189
190    ControlFlow::Continue(())
191}
192
193fn variant_has_complex_ctor(variant: &ty::VariantDef) -> bool {
194    // CtorKind::Const means a "unit" ctor
195    !#[allow(non_exhaustive_omitted_patterns)] match variant.ctor_kind() {
    Some(CtorKind::Const) => true,
    _ => false,
}matches!(variant.ctor_kind(), Some(CtorKind::Const))
196}
197
198/// Per-struct-field function that checks if a struct definition follows
199/// the Power alignment Rule (see the `check_struct_for_power_alignment` function).
200fn check_arg_for_power_alignment<'tcx>(cx: &LateContext<'tcx>, ty: Ty<'tcx>) -> bool {
201    let tcx = cx.tcx;
202    if !(tcx.sess.target.os == Os::Aix) {
    ::core::panicking::panic("assertion failed: tcx.sess.target.os == Os::Aix")
};assert!(tcx.sess.target.os == Os::Aix);
203
204    // Structs (under repr(C)) follow the power alignment rule if:
205    //   - the first field of the struct is a floating-point type that
206    //     is greater than 4-bytes, or
207    //   - the first field of the struct is an aggregate whose
208    //     recursively first field is a floating-point type greater than
209    //     4 bytes.
210    if ty.is_floating_point() && ty.primitive_size(tcx).bytes() > 4 {
211        return true;
212    } else if let Adt(adt_def, _) = ty.kind()
213        && adt_def.is_struct()
214        && adt_def.repr().c()
215        && !adt_def.repr().packed()
216        && adt_def.repr().align.is_none()
217    {
218        let struct_variant = adt_def.variant(VariantIdx::ZERO);
219        // Within a nested struct, all fields are examined to correctly
220        // report if any fields after the nested struct within the
221        // original struct are misaligned.
222        for struct_field in &struct_variant.fields {
223            let field_ty = tcx.type_of(struct_field.did).instantiate_identity().skip_norm_wip();
224            if check_arg_for_power_alignment(cx, field_ty) {
225                return true;
226            }
227        }
228    }
229    return false;
230}
231
232/// Check a struct definition for respect of the Power alignment Rule (as in PowerPC),
233/// which should be respected in the "aix" target OS.
234/// To do so, we must follow one of the two following conditions:
235/// - The first field of the struct must be floating-point type that
236///    is greater than 4-bytes.
237///  - The first field of the struct must be an aggregate whose
238///    recursively first field is a floating-point type greater than
239///    4 bytes.
240fn check_struct_for_power_alignment<'tcx>(
241    cx: &LateContext<'tcx>,
242    item: &'tcx hir::Item<'tcx>,
243    adt_def: AdtDef<'tcx>,
244) {
245    let tcx = cx.tcx;
246
247    // Only consider structs (not enums or unions) on AIX.
248    if tcx.sess.target.os != Os::Aix || !adt_def.is_struct() {
249        return;
250    }
251
252    // The struct must be repr(C), but ignore it if it explicitly specifies its alignment with
253    // either `align(N)` or `packed(N)`.
254    if adt_def.repr().c() && !adt_def.repr().packed() && adt_def.repr().align.is_none() {
255        let struct_variant_data = item.expect_struct().2;
256        for field_def in struct_variant_data.fields().iter().skip(1) {
257            // Struct fields (after the first field) are checked for the
258            // power alignment rule, as fields after the first are likely
259            // to be the fields that are misaligned.
260            let ty = tcx.type_of(field_def.def_id).instantiate_identity().skip_norm_wip();
261            if check_arg_for_power_alignment(cx, ty) {
262                cx.emit_span_lint(USES_POWER_ALIGNMENT, field_def.span, UsesPowerAlignment);
263            }
264        }
265    }
266}
267
268/// Annotates whether we are in the context of an item *defined* in rust
269/// and exposed to an FFI boundary,
270/// or the context of an item from elsewhere, whose interface is re-*declared* in rust.
271#[derive(#[automatically_derived]
impl ::core::clone::Clone for CItemKind {
    #[inline]
    fn clone(&self) -> CItemKind { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for CItemKind { }Copy)]
272enum CItemKind {
273    Declaration,
274    Definition,
275}
276
277/// Annotates whether we are in the context of a function's argument types or return type.
278#[derive(#[automatically_derived]
impl ::core::clone::Clone for FnPos {
    #[inline]
    fn clone(&self) -> FnPos { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for FnPos { }Copy)]
279enum FnPos {
280    Arg,
281    Ret,
282}
283
284enum FfiResult<'tcx> {
285    FfiSafe,
286    FfiPhantom(Ty<'tcx>),
287    FfiUnsafe { ty: Ty<'tcx>, reason: DiagMessage, help: Option<DiagMessage> },
288}
289
290/// The result when a type has been checked but perhaps not completely. `None` indicates that
291/// FFI safety/unsafety has not yet been determined, `Some(res)` indicates that the safety/unsafety
292/// in the `FfiResult` is final.
293type PartialFfiResult<'tcx> = Option<FfiResult<'tcx>>;
294
295/// What type indirection points to a given type.
296#[derive(#[automatically_derived]
impl ::core::clone::Clone for IndirectionKind {
    #[inline]
    fn clone(&self) -> IndirectionKind { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for IndirectionKind { }Copy)]
297enum IndirectionKind {
298    /// Box (valid non-null pointer, owns pointee).
299    Box,
300    /// Ref (valid non-null pointer, borrows pointee).
301    Ref,
302    /// Raw pointer (not necessarily non-null or valid. no info on ownership).
303    RawPtr,
304}
305
306bitflags! {
307    /// VisitorState flags that are linked with the root type's use.
308    /// (These are the permanent part of the state, kept when visiting new Ty.)
309    #[derive(#[automatically_derived]
impl ::core::clone::Clone for RootUseFlags {
    #[inline]
    fn clone(&self) -> RootUseFlags {
        let _:
                ::core::clone::AssertParamIsClone<<RootUseFlags as
                ::bitflags::__private::PublicFlags>::Internal>;
        *self
    }
}
impl RootUseFlags {
    #[doc = r" For use in (externally-linked) static variables."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const STATIC: Self = Self::from_bits_retain(0b000001);
    #[doc = r" For use in functions in general."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const FUNC: Self = Self::from_bits_retain(0b000010);
    #[doc =
    r" For variables in function returns (implicitly: not for static variables)."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const FN_RETURN: Self = Self::from_bits_retain(0b000100);
    #[doc =
    r" For variables in functions/variables which are defined in rust."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const DEFINED: Self = Self::from_bits_retain(0b001000);
    #[doc = r" For times where we are only defining the type of something"]
    #[doc = r" (struct/enum/union definitions, FnPtrs)."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const THEORETICAL: Self = Self::from_bits_retain(0b010000);
}
impl ::bitflags::Flags for RootUseFlags {
    const FLAGS: &'static [::bitflags::Flag<RootUseFlags>] =
        &[{

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("STATIC", RootUseFlags::STATIC)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("FUNC", RootUseFlags::FUNC)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("FN_RETURN", RootUseFlags::FN_RETURN)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("DEFINED", RootUseFlags::DEFINED)
                    },
                    {

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("THEORETICAL",
                            RootUseFlags::THEORETICAL)
                    }];
    type Bits = u8;
    fn bits(&self) -> u8 { RootUseFlags::bits(self) }
    fn from_bits_retain(bits: u8) -> RootUseFlags {
        RootUseFlags::from_bits_retain(bits)
    }
}
#[allow(dead_code, deprecated, unused_doc_comments, unused_attributes,
unused_mut, unused_imports, non_upper_case_globals, clippy ::
assign_op_pattern, clippy :: indexing_slicing, clippy :: same_name_method,
clippy :: iter_without_into_iter,)]
const _: () =
    {
        #[repr(transparent)]
        struct InternalBitFlags(u8);
        #[automatically_derived]
        #[doc(hidden)]
        unsafe impl ::core::clone::TrivialClone for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::clone::Clone for InternalBitFlags {
            #[inline]
            fn clone(&self) -> InternalBitFlags {
                let _: ::core::clone::AssertParamIsClone<u8>;
                *self
            }
        }
        #[automatically_derived]
        impl ::core::marker::Copy for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::marker::StructuralPartialEq for InternalBitFlags { }
        #[automatically_derived]
        impl ::core::cmp::PartialEq for InternalBitFlags {
            #[inline]
            fn eq(&self, other: &InternalBitFlags) -> bool {
                self.0 == other.0
            }
        }
        #[automatically_derived]
        impl ::core::cmp::Eq for InternalBitFlags {
            #[inline]
            #[doc(hidden)]
            #[coverage(off)]
            fn assert_fields_are_eq(&self) {
                let _: ::core::cmp::AssertParamIsEq<u8>;
            }
        }
        #[automatically_derived]
        impl ::core::cmp::PartialOrd for InternalBitFlags {
            #[inline]
            fn partial_cmp(&self, other: &InternalBitFlags)
                -> ::core::option::Option<::core::cmp::Ordering> {
                ::core::option::Option::Some(::core::cmp::Ord::cmp(self,
                        other))
            }
        }
        #[automatically_derived]
        impl ::core::cmp::Ord for InternalBitFlags {
            #[inline]
            fn cmp(&self, other: &InternalBitFlags) -> ::core::cmp::Ordering {
                ::core::cmp::Ord::cmp(&self.0, &other.0)
            }
        }
        #[automatically_derived]
        impl ::core::hash::Hash for InternalBitFlags {
            #[inline]
            fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
                ::core::hash::Hash::hash(&self.0, state)
            }
        }
        impl ::bitflags::__private::PublicFlags for RootUseFlags {
            type Primitive = u8;
            type Internal = InternalBitFlags;
        }
        impl ::bitflags::__private::core::default::Default for
            InternalBitFlags {
            #[inline]
            fn default() -> Self { InternalBitFlags::empty() }
        }
        impl ::bitflags::__private::core::fmt::Debug for InternalBitFlags {
            fn fmt(&self,
                f: &mut ::bitflags::__private::core::fmt::Formatter<'_>)
                -> ::bitflags::__private::core::fmt::Result {
                if self.is_empty() {
                    f.write_fmt(format_args!("{0:#x}",
                            <u8 as ::bitflags::Bits>::EMPTY))
                } else {
                    ::bitflags::__private::core::fmt::Display::fmt(self, f)
                }
            }
        }
        impl ::bitflags::__private::core::fmt::Display for InternalBitFlags {
            fn fmt(&self,
                f: &mut ::bitflags::__private::core::fmt::Formatter<'_>)
                -> ::bitflags::__private::core::fmt::Result {
                ::bitflags::parser::to_writer(&RootUseFlags(*self), f)
            }
        }
        impl ::bitflags::__private::core::str::FromStr for InternalBitFlags {
            type Err = ::bitflags::parser::ParseError;
            fn from_str(s: &str)
                ->
                    ::bitflags::__private::core::result::Result<Self,
                    Self::Err> {
                ::bitflags::parser::from_str::<RootUseFlags>(s).map(|flags|
                        flags.0)
            }
        }
        impl ::bitflags::__private::core::convert::AsRef<u8> for
            InternalBitFlags {
            fn as_ref(&self) -> &u8 { &self.0 }
        }
        impl ::bitflags::__private::core::convert::From<u8> for
            InternalBitFlags {
            fn from(bits: u8) -> Self { Self::from_bits_retain(bits) }
        }
        #[allow(dead_code, deprecated, unused_attributes)]
        impl InternalBitFlags {
            /// Get a flags value with all bits unset.
            #[inline]
            pub const fn empty() -> Self {
                Self(<u8 as ::bitflags::Bits>::EMPTY)
            }
            /// Get a flags value with all known bits set.
            #[inline]
            pub const fn all() -> Self {
                let mut truncated = <u8 as ::bitflags::Bits>::EMPTY;
                let mut i = 0;
                {
                    {
                        let flag =
                            <RootUseFlags as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <RootUseFlags as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <RootUseFlags as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <RootUseFlags as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <RootUseFlags as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                let _ = i;
                Self(truncated)
            }
            /// Get the underlying bits value.
            ///
            /// The returned value is exactly the bits set in this flags value.
            #[inline]
            pub const fn bits(&self) -> u8 { self.0 }
            /// Convert from a bits value.
            ///
            /// This method will return `None` if any unknown bits are set.
            #[inline]
            pub const fn from_bits(bits: u8)
                -> ::bitflags::__private::core::option::Option<Self> {
                let truncated = Self::from_bits_truncate(bits).0;
                if truncated == bits {
                    ::bitflags::__private::core::option::Option::Some(Self(bits))
                } else { ::bitflags::__private::core::option::Option::None }
            }
            /// Convert from a bits value, unsetting any unknown bits.
            #[inline]
            pub const fn from_bits_truncate(bits: u8) -> Self {
                Self(bits & Self::all().0)
            }
            /// Convert from a bits value exactly.
            #[inline]
            pub const fn from_bits_retain(bits: u8) -> Self { Self(bits) }
            /// Get a flags value with the bits of a flag with the given name set.
            ///
            /// This method will return `None` if `name` is empty or doesn't
            /// correspond to any named flag.
            #[inline]
            pub fn from_name(name: &str)
                -> ::bitflags::__private::core::option::Option<Self> {
                {
                    if name == "STATIC" {
                        return ::bitflags::__private::core::option::Option::Some(Self(RootUseFlags::STATIC.bits()));
                    }
                };
                ;
                {
                    if name == "FUNC" {
                        return ::bitflags::__private::core::option::Option::Some(Self(RootUseFlags::FUNC.bits()));
                    }
                };
                ;
                {
                    if name == "FN_RETURN" {
                        return ::bitflags::__private::core::option::Option::Some(Self(RootUseFlags::FN_RETURN.bits()));
                    }
                };
                ;
                {
                    if name == "DEFINED" {
                        return ::bitflags::__private::core::option::Option::Some(Self(RootUseFlags::DEFINED.bits()));
                    }
                };
                ;
                {
                    if name == "THEORETICAL" {
                        return ::bitflags::__private::core::option::Option::Some(Self(RootUseFlags::THEORETICAL.bits()));
                    }
                };
                ;
                let _ = name;
                ::bitflags::__private::core::option::Option::None
            }
            /// Whether all bits in this flags value are unset.
            #[inline]
            pub const fn is_empty(&self) -> bool {
                self.0 == <u8 as ::bitflags::Bits>::EMPTY
            }
            /// Whether all known bits in this flags value are set.
            #[inline]
            pub const fn is_all(&self) -> bool {
                Self::all().0 | self.0 == self.0
            }
            /// Whether any set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn intersects(&self, other: Self) -> bool {
                self.0 & other.0 != <u8 as ::bitflags::Bits>::EMPTY
            }
            /// Whether all set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn contains(&self, other: Self) -> bool {
                self.0 & other.0 == other.0
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            pub fn insert(&mut self, other: Self) {
                *self = Self(self.0).union(other);
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `remove` won't truncate `other`, but the `!` operator will.
            #[inline]
            pub fn remove(&mut self, other: Self) {
                *self = Self(self.0).difference(other);
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            pub fn toggle(&mut self, other: Self) {
                *self = Self(self.0).symmetric_difference(other);
            }
            /// Call `insert` when `value` is `true` or `remove` when `value` is `false`.
            #[inline]
            pub fn set(&mut self, other: Self, value: bool) {
                if value { self.insert(other); } else { self.remove(other); }
            }
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn intersection(self, other: Self) -> Self {
                Self(self.0 & other.0)
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn union(self, other: Self) -> Self {
                Self(self.0 | other.0)
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            #[must_use]
            pub const fn difference(self, other: Self) -> Self {
                Self(self.0 & !other.0)
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn symmetric_difference(self, other: Self) -> Self {
                Self(self.0 ^ other.0)
            }
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            #[must_use]
            pub const fn complement(self) -> Self {
                Self::from_bits_truncate(!self.0)
            }
        }
        impl ::bitflags::__private::core::fmt::Binary for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Binary::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::Octal for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Octal::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::LowerHex for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::LowerHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::UpperHex for InternalBitFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::UpperHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::ops::BitOr for InternalBitFlags {
            type Output = Self;
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor(self, other: InternalBitFlags) -> Self {
                self.union(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitOrAssign for
            InternalBitFlags {
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor_assign(&mut self, other: Self) { self.insert(other); }
        }
        impl ::bitflags::__private::core::ops::BitXor for InternalBitFlags {
            type Output = Self;
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor(self, other: Self) -> Self {
                self.symmetric_difference(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitXorAssign for
            InternalBitFlags {
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor_assign(&mut self, other: Self) { self.toggle(other); }
        }
        impl ::bitflags::__private::core::ops::BitAnd for InternalBitFlags {
            type Output = Self;
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand(self, other: Self) -> Self { self.intersection(other) }
        }
        impl ::bitflags::__private::core::ops::BitAndAssign for
            InternalBitFlags {
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand_assign(&mut self, other: Self) {
                *self =
                    Self::from_bits_retain(self.bits()).intersection(other);
            }
        }
        impl ::bitflags::__private::core::ops::Sub for InternalBitFlags {
            type Output = Self;
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub(self, other: Self) -> Self { self.difference(other) }
        }
        impl ::bitflags::__private::core::ops::SubAssign for InternalBitFlags
            {
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub_assign(&mut self, other: Self) { self.remove(other); }
        }
        impl ::bitflags::__private::core::ops::Not for InternalBitFlags {
            type Output = Self;
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            fn not(self) -> Self { self.complement() }
        }
        impl ::bitflags::__private::core::iter::Extend<InternalBitFlags> for
            InternalBitFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn extend<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(&mut self, iterator: T) {
                for item in iterator { self.insert(item) }
            }
        }
        impl ::bitflags::__private::core::iter::FromIterator<InternalBitFlags>
            for InternalBitFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn from_iter<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(iterator: T) -> Self {
                use ::bitflags::__private::core::iter::Extend;
                let mut result = Self::empty();
                result.extend(iterator);
                result
            }
        }
        impl InternalBitFlags {
            /// Yield a set of contained flags values.
            ///
            /// Each yielded flags value will correspond to a defined named flag. Any unknown bits
            /// will be yielded together as a final flags value.
            #[inline]
            pub const fn iter(&self) -> ::bitflags::iter::Iter<RootUseFlags> {
                ::bitflags::iter::Iter::__private_const_new(<RootUseFlags as
                        ::bitflags::Flags>::FLAGS,
                    RootUseFlags::from_bits_retain(self.bits()),
                    RootUseFlags::from_bits_retain(self.bits()))
            }
            /// Yield a set of contained named flags values.
            ///
            /// This method is like [`iter`](#method.iter), except only yields bits in contained named flags.
            /// Any unknown bits, or bits not corresponding to a contained flag will not be yielded.
            #[inline]
            pub const fn iter_names(&self)
                -> ::bitflags::iter::IterNames<RootUseFlags> {
                ::bitflags::iter::IterNames::__private_const_new(<RootUseFlags
                        as ::bitflags::Flags>::FLAGS,
                    RootUseFlags::from_bits_retain(self.bits()),
                    RootUseFlags::from_bits_retain(self.bits()))
            }
        }
        impl ::bitflags::__private::core::iter::IntoIterator for
            InternalBitFlags {
            type Item = RootUseFlags;
            type IntoIter = ::bitflags::iter::Iter<RootUseFlags>;
            fn into_iter(self) -> Self::IntoIter { self.iter() }
        }
        impl InternalBitFlags {
            /// Returns a mutable reference to the raw value of the flags currently stored.
            #[inline]
            pub fn bits_mut(&mut self) -> &mut u8 { &mut self.0 }
        }
        #[allow(dead_code, deprecated, unused_attributes)]
        impl RootUseFlags {
            /// Get a flags value with all bits unset.
            #[inline]
            pub const fn empty() -> Self { Self(InternalBitFlags::empty()) }
            /// Get a flags value with all known bits set.
            #[inline]
            pub const fn all() -> Self { Self(InternalBitFlags::all()) }
            /// Get the underlying bits value.
            ///
            /// The returned value is exactly the bits set in this flags value.
            #[inline]
            pub const fn bits(&self) -> u8 { self.0.bits() }
            /// Convert from a bits value.
            ///
            /// This method will return `None` if any unknown bits are set.
            #[inline]
            pub const fn from_bits(bits: u8)
                -> ::bitflags::__private::core::option::Option<Self> {
                match InternalBitFlags::from_bits(bits) {
                    ::bitflags::__private::core::option::Option::Some(bits) =>
                        ::bitflags::__private::core::option::Option::Some(Self(bits)),
                    ::bitflags::__private::core::option::Option::None =>
                        ::bitflags::__private::core::option::Option::None,
                }
            }
            /// Convert from a bits value, unsetting any unknown bits.
            #[inline]
            pub const fn from_bits_truncate(bits: u8) -> Self {
                Self(InternalBitFlags::from_bits_truncate(bits))
            }
            /// Convert from a bits value exactly.
            #[inline]
            pub const fn from_bits_retain(bits: u8) -> Self {
                Self(InternalBitFlags::from_bits_retain(bits))
            }
            /// Get a flags value with the bits of a flag with the given name set.
            ///
            /// This method will return `None` if `name` is empty or doesn't
            /// correspond to any named flag.
            #[inline]
            pub fn from_name(name: &str)
                -> ::bitflags::__private::core::option::Option<Self> {
                match InternalBitFlags::from_name(name) {
                    ::bitflags::__private::core::option::Option::Some(bits) =>
                        ::bitflags::__private::core::option::Option::Some(Self(bits)),
                    ::bitflags::__private::core::option::Option::None =>
                        ::bitflags::__private::core::option::Option::None,
                }
            }
            /// Whether all bits in this flags value are unset.
            #[inline]
            pub const fn is_empty(&self) -> bool { self.0.is_empty() }
            /// Whether all known bits in this flags value are set.
            #[inline]
            pub const fn is_all(&self) -> bool { self.0.is_all() }
            /// Whether any set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn intersects(&self, other: Self) -> bool {
                self.0.intersects(other.0)
            }
            /// Whether all set bits in a source flags value are also set in a target flags value.
            #[inline]
            pub const fn contains(&self, other: Self) -> bool {
                self.0.contains(other.0)
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            pub fn insert(&mut self, other: Self) { self.0.insert(other.0) }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `remove` won't truncate `other`, but the `!` operator will.
            #[inline]
            pub fn remove(&mut self, other: Self) { self.0.remove(other.0) }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            pub fn toggle(&mut self, other: Self) { self.0.toggle(other.0) }
            /// Call `insert` when `value` is `true` or `remove` when `value` is `false`.
            #[inline]
            pub fn set(&mut self, other: Self, value: bool) {
                self.0.set(other.0, value)
            }
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn intersection(self, other: Self) -> Self {
                Self(self.0.intersection(other.0))
            }
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn union(self, other: Self) -> Self {
                Self(self.0.union(other.0))
            }
            /// The intersection of a source flags value with the complement of a target flags
            /// value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            #[must_use]
            pub const fn difference(self, other: Self) -> Self {
                Self(self.0.difference(other.0))
            }
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            #[must_use]
            pub const fn symmetric_difference(self, other: Self) -> Self {
                Self(self.0.symmetric_difference(other.0))
            }
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            #[must_use]
            pub const fn complement(self) -> Self {
                Self(self.0.complement())
            }
        }
        impl ::bitflags::__private::core::fmt::Binary for RootUseFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Binary::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::Octal for RootUseFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::Octal::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::LowerHex for RootUseFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::LowerHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::fmt::UpperHex for RootUseFlags {
            fn fmt(&self, f: &mut ::bitflags::__private::core::fmt::Formatter)
                -> ::bitflags::__private::core::fmt::Result {
                let inner = self.0;
                ::bitflags::__private::core::fmt::UpperHex::fmt(&inner, f)
            }
        }
        impl ::bitflags::__private::core::ops::BitOr for RootUseFlags {
            type Output = Self;
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor(self, other: RootUseFlags) -> Self { self.union(other) }
        }
        impl ::bitflags::__private::core::ops::BitOrAssign for RootUseFlags {
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor_assign(&mut self, other: Self) { self.insert(other); }
        }
        impl ::bitflags::__private::core::ops::BitXor for RootUseFlags {
            type Output = Self;
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor(self, other: Self) -> Self {
                self.symmetric_difference(other)
            }
        }
        impl ::bitflags::__private::core::ops::BitXorAssign for RootUseFlags {
            /// The bitwise exclusive-or (`^`) of the bits in two flags values.
            #[inline]
            fn bitxor_assign(&mut self, other: Self) { self.toggle(other); }
        }
        impl ::bitflags::__private::core::ops::BitAnd for RootUseFlags {
            type Output = Self;
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand(self, other: Self) -> Self { self.intersection(other) }
        }
        impl ::bitflags::__private::core::ops::BitAndAssign for RootUseFlags {
            /// The bitwise and (`&`) of the bits in two flags values.
            #[inline]
            fn bitand_assign(&mut self, other: Self) {
                *self =
                    Self::from_bits_retain(self.bits()).intersection(other);
            }
        }
        impl ::bitflags::__private::core::ops::Sub for RootUseFlags {
            type Output = Self;
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub(self, other: Self) -> Self { self.difference(other) }
        }
        impl ::bitflags::__private::core::ops::SubAssign for RootUseFlags {
            /// The intersection of a source flags value with the complement of a target flags value (`&!`).
            ///
            /// This method is not equivalent to `self & !other` when `other` has unknown bits set.
            /// `difference` won't truncate `other`, but the `!` operator will.
            #[inline]
            fn sub_assign(&mut self, other: Self) { self.remove(other); }
        }
        impl ::bitflags::__private::core::ops::Not for RootUseFlags {
            type Output = Self;
            /// The bitwise negation (`!`) of the bits in a flags value, truncating the result.
            #[inline]
            fn not(self) -> Self { self.complement() }
        }
        impl ::bitflags::__private::core::iter::Extend<RootUseFlags> for
            RootUseFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn extend<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(&mut self, iterator: T) {
                for item in iterator { self.insert(item) }
            }
        }
        impl ::bitflags::__private::core::iter::FromIterator<RootUseFlags> for
            RootUseFlags {
            /// The bitwise or (`|`) of the bits in each flags value.
            fn from_iter<T: ::bitflags::__private::core::iter::IntoIterator<Item
                = Self>>(iterator: T) -> Self {
                use ::bitflags::__private::core::iter::Extend;
                let mut result = Self::empty();
                result.extend(iterator);
                result
            }
        }
        impl RootUseFlags {
            /// Yield a set of contained flags values.
            ///
            /// Each yielded flags value will correspond to a defined named flag. Any unknown bits
            /// will be yielded together as a final flags value.
            #[inline]
            pub const fn iter(&self) -> ::bitflags::iter::Iter<RootUseFlags> {
                ::bitflags::iter::Iter::__private_const_new(<RootUseFlags as
                        ::bitflags::Flags>::FLAGS,
                    RootUseFlags::from_bits_retain(self.bits()),
                    RootUseFlags::from_bits_retain(self.bits()))
            }
            /// Yield a set of contained named flags values.
            ///
            /// This method is like [`iter`](#method.iter), except only yields bits in contained named flags.
            /// Any unknown bits, or bits not corresponding to a contained flag will not be yielded.
            #[inline]
            pub const fn iter_names(&self)
                -> ::bitflags::iter::IterNames<RootUseFlags> {
                ::bitflags::iter::IterNames::__private_const_new(<RootUseFlags
                        as ::bitflags::Flags>::FLAGS,
                    RootUseFlags::from_bits_retain(self.bits()),
                    RootUseFlags::from_bits_retain(self.bits()))
            }
        }
        impl ::bitflags::__private::core::iter::IntoIterator for RootUseFlags
            {
            type Item = RootUseFlags;
            type IntoIter = ::bitflags::iter::Iter<RootUseFlags>;
            fn into_iter(self) -> Self::IntoIter { self.iter() }
        }
    };Clone, #[automatically_derived]
impl ::core::marker::Copy for RootUseFlags { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for RootUseFlags {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_tuple_field1_finish(f, "RootUseFlags",
            &&self.0)
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for RootUseFlags {
    #[inline]
    fn eq(&self, other: &RootUseFlags) -> bool { self.0 == other.0 }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for RootUseFlags {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _:
                ::core::cmp::AssertParamIsEq<<RootUseFlags as
                ::bitflags::__private::PublicFlags>::Internal>;
    }
}Eq)]
310    struct RootUseFlags: u8 {
311        /// For use in (externally-linked) static variables.
312        const STATIC = 0b000001;
313        /// For use in functions in general.
314        const FUNC = 0b000010;
315        /// For variables in function returns (implicitly: not for static variables).
316        const FN_RETURN = 0b000100;
317        /// For variables in functions/variables which are defined in rust.
318        const DEFINED = 0b001000;
319        /// For times where we are only defining the type of something
320        /// (struct/enum/union definitions, FnPtrs).
321        const THEORETICAL = 0b010000;
322    }
323}
324
325/// Description of the relationship between current Ty and
326/// the type (or lack thereof) immediately containing it
327#[derive(#[automatically_derived]
impl ::core::marker::Copy for OuterTyKind { }Copy, #[automatically_derived]
impl ::core::clone::Clone for OuterTyKind {
    #[inline]
    fn clone(&self) -> OuterTyKind { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for OuterTyKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                OuterTyKind::None => "None",
                OuterTyKind::NoneThroughFnPtr => "NoneThroughFnPtr",
                OuterTyKind::Other => "Other",
            })
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for OuterTyKind {
    #[inline]
    fn eq(&self, other: &OuterTyKind) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for OuterTyKind {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq)]
328enum OuterTyKind {
329    None,
330    /// A variant that should not exist,
331    /// but is needed because we don't change the lint's behavior yet
332    NoneThroughFnPtr,
333    /// Placeholder for properties that will be used eventually
334    Other,
335}
336
337impl OuterTyKind {
338    /// Computes the relationship by providing the containing Ty itself
339    fn from_ty<'tcx>(ty: Ty<'tcx>) -> Self {
340        match ty.kind() {
341            ty::FnPtr(..) => Self::NoneThroughFnPtr,
342            ty::RawPtr(..)
343            | ty::Ref(..)
344            | ty::Adt(..)
345            | ty::Tuple(..)
346            | ty::Array(..)
347            | ty::Slice(_) => OuterTyKind::Other,
348            _ => ::rustc_middle::util::bug::bug_fmt(format_args!("Unexpected outer type {0:?}",
        ty))bug!("Unexpected outer type {ty:?}"),
349        }
350    }
351}
352
353#[derive(#[automatically_derived]
impl ::core::marker::Copy for VisitorState { }Copy, #[automatically_derived]
impl ::core::clone::Clone for VisitorState {
    #[inline]
    fn clone(&self) -> VisitorState {
        let _: ::core::clone::AssertParamIsClone<RootUseFlags>;
        let _: ::core::clone::AssertParamIsClone<OuterTyKind>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for VisitorState {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field2_finish(f, "VisitorState",
            "root_use_flags", &self.root_use_flags, "outer_ty_kind",
            &&self.outer_ty_kind)
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for VisitorState {
    #[inline]
    fn eq(&self, other: &VisitorState) -> bool {
        self.root_use_flags == other.root_use_flags &&
            self.outer_ty_kind == other.outer_ty_kind
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for VisitorState {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<RootUseFlags>;
        let _: ::core::cmp::AssertParamIsEq<OuterTyKind>;
    }
}Eq)]
354struct VisitorState {
355    /// Flags describing both the overall context in which the current Ty is,
356    /// linked to how the Visitor's original Ty was used.
357    root_use_flags: RootUseFlags,
358    /// Flags describing both the immediate context in which the current Ty is,
359    /// linked to how it relates to its parent Ty (or lack thereof).
360    outer_ty_kind: OuterTyKind,
361}
362
363impl RootUseFlags {
364    // The values that can be set.
365    const STATIC_TY: Self = Self::STATIC;
366    const ARGUMENT_TY_IN_DEFINITION: Self =
367        Self::from_bits(Self::FUNC.bits() | Self::DEFINED.bits()).unwrap();
368    const RETURN_TY_IN_DEFINITION: Self =
369        Self::from_bits(Self::FUNC.bits() | Self::FN_RETURN.bits() | Self::DEFINED.bits()).unwrap();
370    const ARGUMENT_TY_IN_DECLARATION: Self = Self::FUNC;
371    const RETURN_TY_IN_DECLARATION: Self =
372        Self::from_bits(Self::FUNC.bits() | Self::FN_RETURN.bits()).unwrap();
373    const ARGUMENT_TY_IN_FNPTR: Self =
374        Self::from_bits(Self::FUNC.bits() | Self::THEORETICAL.bits()).unwrap();
375    const RETURN_TY_IN_FNPTR: Self =
376        Self::from_bits(Self::FUNC.bits() | Self::THEORETICAL.bits() | Self::FN_RETURN.bits())
377            .unwrap();
378}
379
380impl VisitorState {
381    /// From an existing state, compute the state of any subtype of the current type.
382    /// (General case. For the case where the current type is a function pointer, see `next_in_fnptr`.)
383    fn next(&self, current_ty: Ty<'_>) -> Self {
384        if !!#[allow(non_exhaustive_omitted_patterns)] match current_ty.kind() {
                ty::FnPtr(..) => true,
                _ => false,
            } {
    ::core::panicking::panic("assertion failed: !matches!(current_ty.kind(), ty::FnPtr(..))")
};assert!(!matches!(current_ty.kind(), ty::FnPtr(..)));
385        VisitorState {
386            root_use_flags: self.root_use_flags,
387            outer_ty_kind: OuterTyKind::from_ty(current_ty),
388        }
389    }
390
391    /// From an existing state, compute the state of any subtype of the current type.
392    /// (Case where the current type is a function pointer,
393    /// meaning we need to specify if the subtype is an argument or the return.)
394    fn next_in_fnptr(&self, current_ty: Ty<'_>, fn_pos: FnPos) -> Self {
395        if !#[allow(non_exhaustive_omitted_patterns)] match current_ty.kind() {
            ty::FnPtr(..) => true,
            _ => false,
        } {
    ::core::panicking::panic("assertion failed: matches!(current_ty.kind(), ty::FnPtr(..))")
};assert!(matches!(current_ty.kind(), ty::FnPtr(..)));
396        VisitorState {
397            root_use_flags: match fn_pos {
398                FnPos::Ret => RootUseFlags::RETURN_TY_IN_FNPTR,
399                FnPos::Arg => RootUseFlags::ARGUMENT_TY_IN_FNPTR,
400            },
401            outer_ty_kind: OuterTyKind::from_ty(current_ty),
402        }
403    }
404
405    /// Get the proper visitor state for a given function's arguments or return type.
406    fn fn_entry_point(fn_mode: CItemKind, fn_pos: FnPos) -> Self {
407        let p_flags = match (fn_mode, fn_pos) {
408            (CItemKind::Definition, FnPos::Ret) => RootUseFlags::RETURN_TY_IN_DEFINITION,
409            (CItemKind::Declaration, FnPos::Ret) => RootUseFlags::RETURN_TY_IN_DECLARATION,
410            (CItemKind::Definition, FnPos::Arg) => RootUseFlags::ARGUMENT_TY_IN_DEFINITION,
411            (CItemKind::Declaration, FnPos::Arg) => RootUseFlags::ARGUMENT_TY_IN_DECLARATION,
412        };
413        VisitorState { root_use_flags: p_flags, outer_ty_kind: OuterTyKind::None }
414    }
415
416    /// Get the proper visitor state for a static variable's type
417    fn static_entry_point() -> Self {
418        VisitorState { root_use_flags: RootUseFlags::STATIC_TY, outer_ty_kind: OuterTyKind::None }
419    }
420
421    /// Whether the type is used in a function.
422    fn is_in_function(&self) -> bool {
423        let ret = self.root_use_flags.contains(RootUseFlags::FUNC);
424        if ret {
425            if true {
    if !!self.root_use_flags.contains(RootUseFlags::STATIC) {
        ::core::panicking::panic("assertion failed: !self.root_use_flags.contains(RootUseFlags::STATIC)")
    };
};debug_assert!(!self.root_use_flags.contains(RootUseFlags::STATIC));
426        }
427        ret
428    }
429
430    /// Whether the type is used (directly or not) in a function, in return position.
431    fn is_in_function_return(&self) -> bool {
432        let ret = self.root_use_flags.contains(RootUseFlags::FN_RETURN);
433        if ret {
434            if true {
    if !self.is_in_function() {
        ::core::panicking::panic("assertion failed: self.is_in_function()")
    };
};debug_assert!(self.is_in_function());
435        }
436        ret
437    }
438
439    /// Whether the type is used (directly or not) in a defined function.
440    /// In other words, whether or not we allow non-FFI-safe types behind a C pointer,
441    /// to be treated as an opaque type on the other side of the FFI boundary.
442    fn is_in_defined_function(&self) -> bool {
443        self.root_use_flags.contains(RootUseFlags::DEFINED) && self.is_in_function()
444    }
445
446    /// Whether the type is used (directly or not) in a function pointer type.
447    /// Here, we also allow non-FFI-safe types behind a C pointer,
448    /// to be treated as an opaque type on the other side of the FFI boundary.
449    fn is_in_fnptr(&self) -> bool {
450        self.root_use_flags.contains(RootUseFlags::THEORETICAL) && self.is_in_function()
451    }
452
453    /// Whether we can expect type parameters and co in a given type.
454    fn can_expect_ty_params(&self) -> bool {
455        // rust-defined functions, as well as FnPtrs
456        self.root_use_flags.contains(RootUseFlags::THEORETICAL) || self.is_in_defined_function()
457    }
458}
459
460/// Visitor used to recursively traverse MIR types and evaluate FFI-safety.
461/// It uses ``check_*`` methods as entrypoints to be called elsewhere,
462/// and ``visit_*`` methods to recurse.
463struct ImproperCTypesVisitor<'a, 'tcx> {
464    cx: &'a LateContext<'tcx>,
465    /// To prevent problems with recursive types,
466    /// add a types-in-check cache.
467    cache: FxHashSet<Ty<'tcx>>,
468    /// The original type being checked, before we recursed
469    /// to any other types it contains.
470    base_ty: Ty<'tcx>,
471    base_fn_mode: CItemKind,
472}
473
474impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
475    fn new(
476        cx: &'a LateContext<'tcx>,
477        base_ty: Unnormalized<'tcx, Ty<'tcx>>,
478        base_fn_mode: CItemKind,
479    ) -> Self {
480        // Skip normalization for opaques: even in `TypingMode::Borrowck` the body's own
481        // defining opaques still get revealed, leaving entries in `OpaqueTypeStorage` that
482        // ICE on `InferCtxt` drop (issue #156352).
483        let base_ty = if base_ty.skip_norm_wip().has_opaque_types() {
484            base_ty.skip_norm_wip()
485        } else {
486            maybe_normalize_erasing_regions(cx, base_ty)
487        };
488        ImproperCTypesVisitor { cx, base_ty, base_fn_mode, cache: FxHashSet::default() }
489    }
490
491    /// Checks if the given indirection (box,ref,pointer) is "ffi-safe".
492    fn visit_indirection(
493        &mut self,
494        state: VisitorState,
495        ty: Ty<'tcx>,
496        inner_ty: Ty<'tcx>,
497        indirection_kind: IndirectionKind,
498    ) -> FfiResult<'tcx> {
499        use FfiResult::*;
500        let tcx = self.cx.tcx;
501
502        match indirection_kind {
503            IndirectionKind::Box => {
504                // FIXME(ctypes): this logic is broken, but it still fits the current tests:
505                // - for some reason `Box<_>`es in `extern "ABI" {}` blocks
506                //   (including within FnPtr:s)
507                //   are not treated as pointers but as FFI-unsafe structs
508                // - otherwise, treat the box itself correctly, and follow pointee safety logic
509                //   as described in the other `indirection_type` match branch.
510                if state.is_in_defined_function()
511                    || (state.is_in_fnptr() && #[allow(non_exhaustive_omitted_patterns)] match self.base_fn_mode {
    CItemKind::Definition => true,
    _ => false,
}matches!(self.base_fn_mode, CItemKind::Definition))
512                {
513                    if inner_ty.is_sized(tcx, self.cx.typing_env()) {
514                        return FfiSafe;
515                    } else {
516                        return FfiUnsafe {
517                            ty,
518                            reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("box cannot be represented as a single pointer"))msg!("box cannot be represented as a single pointer"),
519                            help: None,
520                        };
521                    }
522                } else {
523                    // (mid-retcon-commit-chain comment:)
524                    // this is the original fallback behavior, which is wrong
525                    if let ty::Adt(def, args) = ty.kind() {
526                        self.visit_struct_or_union(state, ty, *def, args)
527                    } else if truecfg!(debug_assertions) {
528                        ::rustc_middle::util::bug::bug_fmt(format_args!("ImproperCTypes: this retcon commit was badly written"))bug!("ImproperCTypes: this retcon commit was badly written")
529                    } else {
530                        FfiSafe
531                    }
532                }
533            }
534            IndirectionKind::Ref | IndirectionKind::RawPtr => {
535                // Weird behaviour for pointee safety. the big question here is
536                // "if you have a FFI-unsafe pointee behind a FFI-safe pointer type, is it ok?"
537                // The answer until now is:
538                // "It's OK for rust-defined functions and callbacks, we'll assume those are
539                // meant to be opaque types on the other side of the FFI boundary".
540                //
541                // Reasoning:
542                // For extern function declarations, the actual definition of the function is
543                // written somewhere else, meaning the declaration is free to express this
544                // opaqueness with an extern type (opaque caller-side) or a std::ffi::c_void
545                // (opaque callee-side). For extern function definitions, however, in the case
546                // where the type is opaque caller-side, it is not opaque callee-side,
547                // and having the full type information is necessary to compile the function.
548                //
549                // It might be better to rething this, or even ignore pointee safety for a first
550                // batch of behaviour changes. See the discussion that ends with
551                // https://github.com/rust-lang/rust/pull/134697#issuecomment-2692610258
552                if (state.is_in_defined_function() || state.is_in_fnptr())
553                    && inner_ty.is_sized(self.cx.tcx, self.cx.typing_env())
554                {
555                    FfiSafe
556                } else {
557                    self.visit_type(state.next(ty), inner_ty)
558                }
559            }
560        }
561    }
562
563    /// Checks if the given `VariantDef`'s field types are "ffi-safe".
564    fn visit_variant_fields(
565        &mut self,
566        state: VisitorState,
567        ty: Ty<'tcx>,
568        def: AdtDef<'tcx>,
569        variant: &ty::VariantDef,
570        args: GenericArgsRef<'tcx>,
571    ) -> FfiResult<'tcx> {
572        use FfiResult::*;
573
574        let transparent_with_all_zst_fields = if def.repr().transparent() {
575            if let Some(field) = super::transparent_newtype_field(self.cx.tcx, variant) {
576                // Transparent newtypes have at most one non-ZST field which needs to be checked..
577                let field_ty =
578                    maybe_normalize_erasing_regions(self.cx, field.ty(self.cx.tcx, args));
579                match self.visit_type(state.next(ty), field_ty) {
580                    FfiUnsafe { ty, .. } if ty.is_unit() => (),
581                    r => return r,
582                }
583
584                false
585            } else {
586                // ..or have only ZST fields, which is FFI-unsafe (unless those fields are all
587                // `PhantomData`).
588                true
589            }
590        } else {
591            false
592        };
593
594        // We can't completely trust `repr(C)` markings, so make sure the fields are actually safe.
595        let mut all_phantom = !variant.fields.is_empty();
596        for field in &variant.fields {
597            let field_ty = maybe_normalize_erasing_regions(self.cx, field.ty(self.cx.tcx, args));
598            all_phantom &= match self.visit_type(state.next(ty), field_ty) {
599                FfiSafe => false,
600                // `()` fields are FFI-safe!
601                FfiUnsafe { ty, .. } if ty.is_unit() => false,
602                FfiPhantom(..) => true,
603                r @ FfiUnsafe { .. } => return r,
604            }
605        }
606
607        if all_phantom {
608            FfiPhantom(ty)
609        } else if transparent_with_all_zst_fields {
610            FfiUnsafe {
611                ty,
612                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this struct contains only zero-sized fields"))msg!("this struct contains only zero-sized fields"),
613                help: None,
614            }
615        } else {
616            FfiSafe
617        }
618    }
619
620    fn visit_struct_or_union(
621        &mut self,
622        state: VisitorState,
623        ty: Ty<'tcx>,
624        def: AdtDef<'tcx>,
625        args: GenericArgsRef<'tcx>,
626    ) -> FfiResult<'tcx> {
627        if true {
    if !#[allow(non_exhaustive_omitted_patterns)] match def.adt_kind() {
                AdtKind::Struct | AdtKind::Union => true,
                _ => false,
            } {
        ::core::panicking::panic("assertion failed: matches!(def.adt_kind(), AdtKind::Struct | AdtKind::Union)")
    };
};debug_assert!(matches!(def.adt_kind(), AdtKind::Struct | AdtKind::Union));
628        use FfiResult::*;
629
630        if !def.repr().c() && !def.repr().transparent() {
631            return FfiUnsafe {
632                ty,
633                reason: if def.is_struct() {
634                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this struct has unspecified layout"))msg!("this struct has unspecified layout")
635                } else {
636                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this union has unspecified layout"))msg!("this union has unspecified layout")
637                },
638                help: if def.is_struct() {
639                    Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider adding a `#[repr(C)]` or `#[repr(transparent)]` attribute to this struct"))msg!(
640                        "consider adding a `#[repr(C)]` or `#[repr(transparent)]` attribute to this struct"
641                    ))
642                } else {
643                    // FIXME(#60405): confirm that this makes sense for unions once #60405 / RFC2645 stabilises
644                    Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider adding a `#[repr(C)]` or `#[repr(transparent)]` attribute to this union"))msg!(
645                        "consider adding a `#[repr(C)]` or `#[repr(transparent)]` attribute to this union"
646                    ))
647                },
648            };
649        }
650
651        if def.non_enum_variant().field_list_has_applicable_non_exhaustive() {
652            return FfiUnsafe {
653                ty,
654                reason: if def.is_struct() {
655                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this struct is non-exhaustive"))msg!("this struct is non-exhaustive")
656                } else {
657                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this union is non-exhaustive"))msg!("this union is non-exhaustive")
658                },
659                help: None,
660            };
661        }
662
663        if def.non_enum_variant().fields.is_empty() {
664            FfiUnsafe {
665                ty,
666                reason: if def.is_struct() {
667                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this struct has no fields"))msg!("this struct has no fields")
668                } else {
669                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this union has no fields"))msg!("this union has no fields")
670                },
671                help: if def.is_struct() {
672                    Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider adding a member to this struct"))msg!("consider adding a member to this struct"))
673                } else {
674                    Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider adding a member to this union"))msg!("consider adding a member to this union"))
675                },
676            }
677        } else {
678            self.visit_variant_fields(state, ty, def, def.non_enum_variant(), args)
679        }
680    }
681
682    fn visit_enum(
683        &mut self,
684        state: VisitorState,
685        ty: Ty<'tcx>,
686        def: AdtDef<'tcx>,
687        args: GenericArgsRef<'tcx>,
688    ) -> FfiResult<'tcx> {
689        if true {
    if !#[allow(non_exhaustive_omitted_patterns)] match def.adt_kind() {
                AdtKind::Enum => true,
                _ => false,
            } {
        ::core::panicking::panic("assertion failed: matches!(def.adt_kind(), AdtKind::Enum)")
    };
};debug_assert!(matches!(def.adt_kind(), AdtKind::Enum));
690        use FfiResult::*;
691
692        if def.variants().is_empty() {
693            // Empty enums are okay... although sort of useless.
694            return FfiSafe;
695        }
696        // Check for a repr() attribute to specify the size of the
697        // discriminant.
698        if !def.repr().c() && !def.repr().transparent() && def.repr().int.is_none() {
699            // Special-case types like `Option<extern fn()>` and `Result<extern fn(), ()>`
700            if let Some(inner_ty) = repr_nullable_ptr(self.cx.tcx, self.cx.typing_env(), ty) {
701                return self.visit_type(state.next(ty), inner_ty);
702            }
703
704            return FfiUnsafe {
705                ty,
706                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("enum has no representation hint"))msg!("enum has no representation hint"),
707                help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider adding a `#[repr(C)]`, `#[repr(transparent)]`, or integer `#[repr(...)]` attribute to this enum"))msg!(
708                    "consider adding a `#[repr(C)]`, `#[repr(transparent)]`, or integer `#[repr(...)]` attribute to this enum"
709                )),
710            };
711        }
712
713        let non_exhaustive = def.variant_list_has_applicable_non_exhaustive();
714        // Check the contained variants.
715        let ret = def.variants().iter().try_for_each(|variant| {
716            check_non_exhaustive_variant(non_exhaustive, variant)
717                .map_break(|reason| FfiUnsafe { ty, reason, help: None })?;
718
719            match self.visit_variant_fields(state, ty, def, variant, args) {
720                FfiSafe => ControlFlow::Continue(()),
721                r => ControlFlow::Break(r),
722            }
723        });
724        if let ControlFlow::Break(result) = ret {
725            return result;
726        }
727
728        FfiSafe
729    }
730
731    /// Checks if the given type is "ffi-safe" (has a stable, well-defined
732    /// representation which can be exported to C code).
733    fn visit_type(&mut self, state: VisitorState, ty: Ty<'tcx>) -> FfiResult<'tcx> {
734        use FfiResult::*;
735
736        let tcx = self.cx.tcx;
737
738        // Protect against infinite recursion, for example
739        // `struct S(*mut S);`.
740        // FIXME: A recursion limit is necessary as well, for irregular
741        // recursive types.
742        if !self.cache.insert(ty) {
743            return FfiSafe;
744        }
745
746        match *ty.kind() {
747            ty::Adt(def, args) => {
748                if let Some(inner_ty) = ty.boxed_ty() {
749                    return self.visit_indirection(state, ty, inner_ty, IndirectionKind::Box);
750                }
751                if def.is_phantom_data() {
752                    return FfiPhantom(ty);
753                }
754                match def.adt_kind() {
755                    AdtKind::Struct | AdtKind::Union => {
756                        if let Some(sym::cstring_type | sym::cstr_type) =
757                            tcx.get_diagnostic_name(def.did())
758                            && !self.base_ty.is_mutable_ptr()
759                        {
760                            return FfiUnsafe {
761                                ty,
762                                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("`CStr`/`CString` do not have a guaranteed layout"))msg!("`CStr`/`CString` do not have a guaranteed layout"),
763                                help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider passing a `*const std::ffi::c_char` instead, and use `CStr::as_ptr()`"))msg!(
764                                    "consider passing a `*const std::ffi::c_char` instead, and use `CStr::as_ptr()`"
765                                )),
766                            };
767                        }
768                        self.visit_struct_or_union(state, ty, def, args)
769                    }
770                    AdtKind::Enum => self.visit_enum(state, ty, def, args),
771                }
772            }
773
774            // Pattern types are just extra invariants on the type that you need to uphold,
775            // but only the base type is relevant for being representable in FFI.
776            // (note: this lint was written when pattern types could only be integers constrained to ranges)
777            // (also note: the lack of ".next(ty)" on the state is on purpose)
778            ty::Pat(pat_ty, _) => self.visit_type(state, pat_ty),
779
780            // types which likely have a stable representation, if the target architecture defines those
781            // note: before rust 1.77, 128-bit ints were not FFI-safe on x86_64
782            ty::Int(..) | ty::Uint(..) | ty::Float(..) => FfiResult::FfiSafe,
783
784            ty::Bool => FfiResult::FfiSafe,
785
786            ty::Char => FfiResult::FfiUnsafe {
787                ty,
788                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("the `char` type has no C equivalent"))msg!("the `char` type has no C equivalent"),
789                help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using `u32` or `libc::wchar_t` instead"))msg!("consider using `u32` or `libc::wchar_t` instead")),
790            },
791
792            ty::Slice(_) => FfiUnsafe {
793                ty,
794                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("slices have no C equivalent"))msg!("slices have no C equivalent"),
795                help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using a raw pointer instead"))msg!("consider using a raw pointer instead")),
796            },
797
798            ty::Dynamic(..) => {
799                FfiUnsafe { ty, reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("trait objects have no C equivalent"))msg!("trait objects have no C equivalent"), help: None }
800            }
801
802            ty::Str => FfiUnsafe {
803                ty,
804                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("string slices have no C equivalent"))msg!("string slices have no C equivalent"),
805                help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using `*const u8` and a length instead"))msg!("consider using `*const u8` and a length instead")),
806            },
807
808            ty::Tuple(tuple) => {
809                if tuple.is_empty()
810                    && state.is_in_function_return()
811                    && #[allow(non_exhaustive_omitted_patterns)] match state.outer_ty_kind {
    OuterTyKind::None | OuterTyKind::NoneThroughFnPtr => true,
    _ => false,
}matches!(
812                        state.outer_ty_kind,
813                        OuterTyKind::None | OuterTyKind::NoneThroughFnPtr
814                    )
815                {
816                    // C functions can return void
817                    FfiSafe
818                } else {
819                    FfiUnsafe {
820                        ty,
821                        reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("tuples have unspecified layout"))msg!("tuples have unspecified layout"),
822                        help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using a struct instead"))msg!("consider using a struct instead")),
823                    }
824                }
825            }
826
827            ty::RawPtr(ty, _)
828                if match ty.kind() {
829                    ty::Tuple(tuple) => tuple.is_empty(),
830                    _ => false,
831                } =>
832            {
833                FfiSafe
834            }
835
836            ty::RawPtr(inner_ty, _) => {
837                return self.visit_indirection(state, ty, inner_ty, IndirectionKind::RawPtr);
838            }
839            ty::Ref(_, inner_ty, _) => {
840                return self.visit_indirection(state, ty, inner_ty, IndirectionKind::Ref);
841            }
842
843            ty::Array(inner_ty, _) => {
844                if state.is_in_function()
845                    // FIXME(ctypes): VVV-this-VVV shouldn't make a difference between ::None and ::NoneThroughFnPtr
846                    && #[allow(non_exhaustive_omitted_patterns)] match state.outer_ty_kind {
    OuterTyKind::None => true,
    _ => false,
}matches!(state.outer_ty_kind, OuterTyKind::None)
847                {
848                    // C doesn't really support passing arrays by value - the only way to pass an array by value
849                    // is through a struct.
850                    FfiResult::FfiUnsafe {
851                        ty,
852                        reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("passing raw arrays by value is not FFI-safe"))msg!("passing raw arrays by value is not FFI-safe"),
853                        help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider passing a pointer to the array"))msg!("consider passing a pointer to the array")),
854                    }
855                } else {
856                    // let's allow phantoms to go through,
857                    // since an array of 1-ZSTs is also a 1-ZST
858                    self.visit_type(state.next(ty), inner_ty)
859                }
860            }
861
862            ty::FnPtr(sig_tys, hdr) => {
863                let sig = sig_tys.with(hdr);
864                if sig.abi().is_rustic_abi() {
865                    return FfiUnsafe {
866                        ty,
867                        reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("this function pointer has Rust-specific calling convention"))msg!("this function pointer has Rust-specific calling convention"),
868                        help: Some(rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("consider using an `extern fn(...) -> ...` function pointer instead"))msg!(
869                            "consider using an `extern fn(...) -> ...` function pointer instead"
870                        )),
871                    };
872                }
873
874                let sig = tcx.instantiate_bound_regions_with_erased(sig);
875                for arg in sig.inputs() {
876                    match self.visit_type(state.next_in_fnptr(ty, FnPos::Arg), *arg) {
877                        FfiSafe => {}
878                        r => return r,
879                    }
880                }
881
882                let ret_ty = sig.output();
883                self.visit_type(state.next_in_fnptr(ty, FnPos::Ret), ret_ty)
884            }
885
886            ty::Foreign(..) => FfiSafe,
887
888            ty::Never => FfiSafe,
889
890            // While opaque types are checked for earlier, if a projection in a struct field
891            // normalizes to an opaque type, then it will reach this branch.
892            ty::Alias(ty::AliasTy { kind: ty::Opaque { .. }, .. }) => {
893                FfiUnsafe { ty, reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("opaque types have no C equivalent"))msg!("opaque types have no C equivalent"), help: None }
894            }
895
896            // `extern "C" fn` functions can have type parameters, which may or may not be FFI-safe,
897            //  so they are currently ignored for the purposes of this lint.
898            ty::Param(..)
899            | ty::Alias(ty::AliasTy {
900                kind: ty::Projection { .. } | ty::Inherent { .. }, ..
901            }) if state.can_expect_ty_params() => FfiSafe,
902
903            ty::UnsafeBinder(_) => FfiUnsafe {
904                ty,
905                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("unsafe binders are incompatible with foreign function interfaces"))msg!("unsafe binders are incompatible with foreign function interfaces"),
906                help: None,
907            },
908
909            // Safety net for when normalization reveals a body's own defining opaque
910            // (e.g. `async extern fn`'s `impl Future` → `Coroutine`); the nicer
911            // "opaque types have no C equivalent" message comes from `visit_for_opaque_ty`
912            // in `check_type` before normalization (issue #156352).
913            ty::Closure(..)
914            | ty::CoroutineClosure(..)
915            | ty::Coroutine(..)
916            | ty::CoroutineWitness(..) => FfiUnsafe {
917                ty,
918                reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("closures and coroutines are not FFI-safe"))msg!("closures and coroutines are not FFI-safe"),
919                help: None,
920            },
921
922            ty::Param(..)
923            | ty::Alias(ty::AliasTy {
924                kind: ty::Projection { .. } | ty::Inherent { .. } | ty::Free { .. },
925                ..
926            })
927            | ty::Infer(..)
928            | ty::Bound(..)
929            | ty::Error(_)
930            | ty::Placeholder(..)
931            | ty::FnDef(..) => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected type in foreign function: {0:?}",
        ty))bug!("unexpected type in foreign function: {:?}", ty),
932        }
933    }
934
935    fn visit_for_opaque_ty(&mut self, ty: Ty<'tcx>) -> PartialFfiResult<'tcx> {
936        struct ProhibitOpaqueTypes;
937        impl<'tcx> ty::TypeVisitor<TyCtxt<'tcx>> for ProhibitOpaqueTypes {
938            type Result = ControlFlow<Ty<'tcx>>;
939
940            fn visit_ty(&mut self, ty: Ty<'tcx>) -> Self::Result {
941                if !ty.has_opaque_types() {
942                    return ControlFlow::Continue(());
943                }
944
945                if let ty::Alias(ty::AliasTy { kind: ty::Opaque { .. }, .. }) = ty.kind() {
946                    ControlFlow::Break(ty)
947                } else {
948                    ty.super_visit_with(self)
949                }
950            }
951        }
952
953        ty.visit_with(&mut ProhibitOpaqueTypes).break_value().map(|ty| FfiResult::FfiUnsafe {
954            ty,
955            reason: rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("opaque types have no C equivalent"))msg!("opaque types have no C equivalent"),
956            help: None,
957        })
958    }
959
960    fn check_type(
961        &mut self,
962        state: VisitorState,
963        ty: Unnormalized<'tcx, Ty<'tcx>>,
964    ) -> FfiResult<'tcx> {
965        // Catch opaques before normalization so the new solver doesn't reveal them
966        // (e.g. `async extern fn` return → `Coroutine`) and we get the nicer
967        // "opaque types have no C equivalent" message.
968        if let Some(res) = self.visit_for_opaque_ty(ty.skip_norm_wip()) {
969            return res;
970        }
971        let ty = maybe_normalize_erasing_regions(self.cx, ty);
972        if let Some(res) = self.visit_for_opaque_ty(ty) {
973            return res;
974        }
975
976        self.visit_type(state, ty)
977    }
978}
979
980impl<'tcx> ImproperCTypesLint {
981    /// Find any fn-ptr types with external ABIs in `ty`, and FFI-checks them.
982    /// For example, `Option<extern "C" fn()>` FFI-checks `extern "C" fn()`.
983    fn check_type_for_external_abi_fnptr(
984        &mut self,
985        cx: &LateContext<'tcx>,
986        state: VisitorState,
987        hir_ty: &hir::Ty<'tcx>,
988        ty: Ty<'tcx>,
989        fn_mode: CItemKind,
990    ) {
991        struct FnPtrFinder<'tcx> {
992            spans: Vec<Span>,
993            tys: Vec<Ty<'tcx>>,
994        }
995
996        impl<'tcx> hir::intravisit::Visitor<'_> for FnPtrFinder<'tcx> {
997            fn visit_ty(&mut self, ty: &'_ hir::Ty<'_, AmbigArg>) {
998                {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_lint/src/types/improper_ctypes.rs:998",
                        "rustc_lint::types::improper_ctypes",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_lint/src/types/improper_ctypes.rs"),
                        ::tracing_core::__macro_support::Option::Some(998u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_lint::types::improper_ctypes"),
                        ::tracing_core::field::FieldSet::new(&["ty"],
                            ::tracing_core::callsite::Identifier(&__CALLSITE)),
                        ::tracing::metadata::Kind::EVENT)
                };
            ::tracing::callsite::DefaultCallsite::new(&META)
        };
    let enabled =
        ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() &&
            {
                let interest = __CALLSITE.interest();
                !interest.is_never() &&
                    ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                        interest)
            };
    if enabled {
        (|value_set: ::tracing::field::ValueSet|
                    {
                        let meta = __CALLSITE.metadata();
                        ::tracing::Event::dispatch(meta, &value_set);
                        ;
                    })({
                #[allow(unused_imports)]
                use ::tracing::field::{debug, display, Value};
                let mut iter = __CALLSITE.metadata().fields().iter();
                __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                    ::tracing::__macro_support::Option::Some(&debug(&ty) as
                                            &dyn Value))])
            });
    } else { ; }
};debug!(?ty);
999                if let hir::TyKind::FnPtr(hir::FnPtrTy { abi, .. }) = ty.kind
1000                    && !abi.is_rustic_abi()
1001                {
1002                    self.spans.push(ty.span);
1003                }
1004
1005                hir::intravisit::walk_ty(self, ty);
1006            }
1007        }
1008
1009        impl<'tcx> ty::TypeVisitor<TyCtxt<'tcx>> for FnPtrFinder<'tcx> {
1010            type Result = ();
1011
1012            fn visit_ty(&mut self, ty: Ty<'tcx>) -> Self::Result {
1013                if let ty::FnPtr(_, hdr) = ty.kind()
1014                    && !hdr.abi().is_rustic_abi()
1015                {
1016                    self.tys.push(ty);
1017                }
1018
1019                ty.super_visit_with(self)
1020            }
1021        }
1022
1023        let mut visitor = FnPtrFinder { spans: Vec::new(), tys: Vec::new() };
1024        ty.visit_with(&mut visitor);
1025        visitor.visit_ty_unambig(hir_ty);
1026
1027        let all_types = iter::zip(visitor.tys.drain(..), visitor.spans.drain(..));
1028        for (fn_ptr_ty, span) in all_types {
1029            let fn_ptr_ty = Unnormalized::new_wip(fn_ptr_ty);
1030            let mut visitor = ImproperCTypesVisitor::new(cx, fn_ptr_ty, fn_mode);
1031            // FIXME(ctypes): make a check_for_fnptr
1032            let ffi_res = visitor.check_type(state, fn_ptr_ty);
1033
1034            self.process_ffi_result(cx, span, ffi_res, fn_mode);
1035        }
1036    }
1037
1038    /// Regardless of a function's need to be "ffi-safe", look for fn-ptr argument/return types
1039    /// that need to be checked for ffi-safety.
1040    fn check_fn_for_external_abi_fnptr(
1041        &mut self,
1042        cx: &LateContext<'tcx>,
1043        fn_mode: CItemKind,
1044        def_id: LocalDefId,
1045        decl: &'tcx hir::FnDecl<'_>,
1046    ) {
1047        let sig = cx.tcx.fn_sig(def_id).instantiate_identity().skip_norm_wip();
1048        let sig = cx.tcx.instantiate_bound_regions_with_erased(sig);
1049
1050        for (input_ty, input_hir) in iter::zip(sig.inputs(), decl.inputs) {
1051            let state = VisitorState::fn_entry_point(fn_mode, FnPos::Arg);
1052            self.check_type_for_external_abi_fnptr(cx, state, input_hir, *input_ty, fn_mode);
1053        }
1054
1055        if let hir::FnRetTy::Return(ret_hir) = decl.output {
1056            let state = VisitorState::fn_entry_point(fn_mode, FnPos::Ret);
1057            self.check_type_for_external_abi_fnptr(cx, state, ret_hir, sig.output(), fn_mode);
1058        }
1059    }
1060
1061    /// For a local definition of a #[repr(C)] struct/enum/union, check that it is indeed FFI-safe.
1062    fn check_reprc_adt(
1063        &mut self,
1064        cx: &LateContext<'tcx>,
1065        item: &'tcx hir::Item<'tcx>,
1066        adt_def: AdtDef<'tcx>,
1067    ) {
1068        if true {
    if !(adt_def.repr().c() && !adt_def.repr().packed() &&
                adt_def.repr().align.is_none()) {
        ::core::panicking::panic("assertion failed: adt_def.repr().c() && !adt_def.repr().packed() &&\n    adt_def.repr().align.is_none()")
    };
};debug_assert!(
1069            adt_def.repr().c() && !adt_def.repr().packed() && adt_def.repr().align.is_none()
1070        );
1071
1072        // FIXME(ctypes): this following call is awkward.
1073        // is there a way to perform its logic in MIR space rather than HIR space?
1074        // (so that its logic can be absorbed into visitor.visit_struct_or_union)
1075        check_struct_for_power_alignment(cx, item, adt_def);
1076    }
1077
1078    fn check_foreign_static(&mut self, cx: &LateContext<'tcx>, id: hir::OwnerId, span: Span) {
1079        let ty = cx.tcx.type_of(id).instantiate_identity();
1080        let mut visitor = ImproperCTypesVisitor::new(cx, ty, CItemKind::Declaration);
1081        let ffi_res = visitor.check_type(VisitorState::static_entry_point(), ty);
1082        self.process_ffi_result(cx, span, ffi_res, CItemKind::Declaration);
1083    }
1084
1085    /// Check if a function's argument types and result type are "ffi-safe".
1086    fn check_foreign_fn(
1087        &mut self,
1088        cx: &LateContext<'tcx>,
1089        fn_mode: CItemKind,
1090        def_id: LocalDefId,
1091        decl: &'tcx hir::FnDecl<'_>,
1092    ) {
1093        let sig = cx.tcx.fn_sig(def_id).instantiate_identity().skip_norm_wip();
1094        let sig = cx.tcx.instantiate_bound_regions_with_erased(sig);
1095
1096        for (input_ty, input_hir) in iter::zip(sig.inputs(), decl.inputs) {
1097            let input_ty = Unnormalized::new_wip(*input_ty);
1098            let state = VisitorState::fn_entry_point(fn_mode, FnPos::Arg);
1099            let mut visitor = ImproperCTypesVisitor::new(cx, input_ty, fn_mode);
1100            let ffi_res = visitor.check_type(state, input_ty);
1101            self.process_ffi_result(cx, input_hir.span, ffi_res, fn_mode);
1102        }
1103
1104        if let hir::FnRetTy::Return(ret_hir) = decl.output {
1105            let output_ty = Unnormalized::new_wip(sig.output());
1106            let state = VisitorState::fn_entry_point(fn_mode, FnPos::Ret);
1107            let mut visitor = ImproperCTypesVisitor::new(cx, output_ty, fn_mode);
1108            let ffi_res = visitor.check_type(state, output_ty);
1109            self.process_ffi_result(cx, ret_hir.span, ffi_res, fn_mode);
1110        }
1111    }
1112
1113    fn process_ffi_result(
1114        &self,
1115        cx: &LateContext<'tcx>,
1116        sp: Span,
1117        res: FfiResult<'tcx>,
1118        fn_mode: CItemKind,
1119    ) {
1120        match res {
1121            FfiResult::FfiSafe => {}
1122            FfiResult::FfiPhantom(ty) => {
1123                self.emit_ffi_unsafe_type_lint(
1124                    cx,
1125                    ty,
1126                    sp,
1127                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("composed only of `PhantomData`"))msg!("composed only of `PhantomData`"),
1128                    None,
1129                    fn_mode,
1130                );
1131            }
1132            FfiResult::FfiUnsafe { ty, reason, help } => {
1133                self.emit_ffi_unsafe_type_lint(cx, ty, sp, reason, help, fn_mode);
1134            }
1135        }
1136    }
1137
1138    fn emit_ffi_unsafe_type_lint(
1139        &self,
1140        cx: &LateContext<'tcx>,
1141        ty: Ty<'tcx>,
1142        sp: Span,
1143        note: DiagMessage,
1144        help: Option<DiagMessage>,
1145        fn_mode: CItemKind,
1146    ) {
1147        let lint = match fn_mode {
1148            CItemKind::Declaration => IMPROPER_CTYPES,
1149            CItemKind::Definition => IMPROPER_CTYPES_DEFINITIONS,
1150        };
1151        let desc = match fn_mode {
1152            CItemKind::Declaration => "block",
1153            CItemKind::Definition => "fn",
1154        };
1155        let span_note = if let ty::Adt(def, _) = ty.kind()
1156            && let Some(sp) = cx.tcx.hir_span_if_local(def.did())
1157        {
1158            Some(sp)
1159        } else {
1160            None
1161        };
1162        cx.emit_span_lint(lint, sp, ImproperCTypes { ty, desc, label: sp, help, note, span_note });
1163    }
1164}
1165
1166/// `ImproperCTypesDefinitions` checks items outside of foreign items (e.g. stuff that isn't in
1167/// `extern "C" { }` blocks):
1168///
1169/// - `extern "<abi>" fn` definitions are checked in the same way as the
1170///   `ImproperCtypesDeclarations` visitor checks functions if `<abi>` is external (e.g. "C").
1171/// - All other items which contain types (e.g. other functions, struct definitions, etc) are
1172///   checked for extern fn-ptrs with external ABIs.
1173impl<'tcx> LateLintPass<'tcx> for ImproperCTypesLint {
1174    fn check_foreign_item(&mut self, cx: &LateContext<'tcx>, it: &hir::ForeignItem<'tcx>) {
1175        let abi = cx.tcx.hir_get_foreign_abi(it.hir_id());
1176
1177        match it.kind {
1178            hir::ForeignItemKind::Fn(sig, _, _) => {
1179                // fnptrs are a special case, they always need to be treated as
1180                // "the element rendered unsafe" because their unsafety doesn't affect
1181                // their surroundings, and their type is often declared inline
1182                if !abi.is_rustic_abi() {
1183                    self.check_foreign_fn(cx, CItemKind::Declaration, it.owner_id.def_id, sig.decl);
1184                } else {
1185                    self.check_fn_for_external_abi_fnptr(
1186                        cx,
1187                        CItemKind::Declaration,
1188                        it.owner_id.def_id,
1189                        sig.decl,
1190                    );
1191                }
1192            }
1193            hir::ForeignItemKind::Static(ty, _, _) if !abi.is_rustic_abi() => {
1194                self.check_foreign_static(cx, it.owner_id, ty.span);
1195            }
1196            hir::ForeignItemKind::Static(..) | hir::ForeignItemKind::Type => (),
1197        }
1198    }
1199
1200    fn check_item(&mut self, cx: &LateContext<'tcx>, item: &'tcx hir::Item<'tcx>) {
1201        match item.kind {
1202            hir::ItemKind::Static(_, _, ty, _)
1203            | hir::ItemKind::Const(_, _, ty, _)
1204            | hir::ItemKind::TyAlias(_, _, ty) => {
1205                self.check_type_for_external_abi_fnptr(
1206                    cx,
1207                    VisitorState::static_entry_point(),
1208                    ty,
1209                    cx.tcx.type_of(item.owner_id).instantiate_identity().skip_norm_wip(),
1210                    CItemKind::Definition,
1211                );
1212            }
1213            // See `check_fn` for declarations, `check_foreign_items` for definitions in extern blocks
1214            hir::ItemKind::Fn { .. } => {}
1215            hir::ItemKind::Struct(..) | hir::ItemKind::Union(..) | hir::ItemKind::Enum(..) => {
1216                // looking for extern FnPtr:s is delegated to `check_field_def`.
1217                let adt_def: AdtDef<'tcx> = cx.tcx.adt_def(item.owner_id.to_def_id());
1218
1219                if adt_def.repr().c() && !adt_def.repr().packed() && adt_def.repr().align.is_none()
1220                {
1221                    self.check_reprc_adt(cx, item, adt_def);
1222                }
1223            }
1224
1225            // Doesn't define something that can contain a external type to be checked.
1226            hir::ItemKind::Impl(..)
1227            | hir::ItemKind::TraitAlias(..)
1228            | hir::ItemKind::Trait { .. }
1229            | hir::ItemKind::GlobalAsm { .. }
1230            | hir::ItemKind::ForeignMod { .. }
1231            | hir::ItemKind::Mod(..)
1232            | hir::ItemKind::Macro(..)
1233            | hir::ItemKind::Use(..)
1234            | hir::ItemKind::ExternCrate(..) => {}
1235        }
1236    }
1237
1238    fn check_field_def(&mut self, cx: &LateContext<'tcx>, field: &'tcx hir::FieldDef<'tcx>) {
1239        self.check_type_for_external_abi_fnptr(
1240            cx,
1241            VisitorState::static_entry_point(),
1242            field.ty,
1243            cx.tcx.type_of(field.def_id).instantiate_identity().skip_norm_wip(),
1244            CItemKind::Definition,
1245        );
1246    }
1247
1248    fn check_fn(
1249        &mut self,
1250        cx: &LateContext<'tcx>,
1251        kind: hir::intravisit::FnKind<'tcx>,
1252        decl: &'tcx hir::FnDecl<'_>,
1253        _: &'tcx hir::Body<'_>,
1254        _: Span,
1255        id: LocalDefId,
1256    ) {
1257        use hir::intravisit::FnKind;
1258
1259        let abi = match kind {
1260            FnKind::ItemFn(_, _, header, ..) => header.abi,
1261            FnKind::Method(_, sig, ..) => sig.header.abi,
1262            _ => return,
1263        };
1264
1265        // fnptrs are a special case, they always need to be treated as
1266        // "the element rendered unsafe" because their unsafety doesn't affect
1267        // their surroundings, and their type is often declared inline
1268        if !abi.is_rustic_abi() {
1269            self.check_foreign_fn(cx, CItemKind::Definition, id, decl);
1270        } else {
1271            self.check_fn_for_external_abi_fnptr(cx, CItemKind::Definition, id, decl);
1272        }
1273    }
1274}