Skip to main content

rustc_target/callconv/
mod.rs

1use std::{fmt, iter};
2
3use arrayvec::ArrayVec;
4use rustc_abi::{
5    AddressSpace, Align, BackendRepr, CanonAbi, ExternAbi, FieldsShape, HasDataLayout, Primitive,
6    Reg, RegKind, Scalar, Size, TyAbiInterface, TyAndLayout, Variants,
7};
8use rustc_macros::StableHash;
9
10pub use crate::spec::AbiMap;
11use crate::spec::{Arch, HasTargetSpec, HasX86AbiOpt};
12
13mod aarch64;
14mod amdgpu;
15mod arm;
16mod avr;
17mod bpf;
18mod csky;
19mod hexagon;
20mod loongarch;
21mod m68k;
22mod mips;
23mod mips64;
24mod msp430;
25mod nvptx64;
26mod powerpc;
27mod powerpc64;
28mod riscv;
29mod s390x;
30mod sparc;
31mod sparc64;
32mod wasm;
33mod x86;
34mod x86_64;
35mod x86_win32;
36mod x86_win64;
37mod xtensa;
38
39#[derive(#[automatically_derived]
impl ::core::clone::Clone for PassMode {
    #[inline]
    fn clone(&self) -> PassMode {
        match self {
            PassMode::Ignore => PassMode::Ignore,
            PassMode::Direct(__self_0) =>
                PassMode::Direct(::core::clone::Clone::clone(__self_0)),
            PassMode::Pair(__self_0, __self_1) =>
                PassMode::Pair(::core::clone::Clone::clone(__self_0),
                    ::core::clone::Clone::clone(__self_1)),
            PassMode::Cast { pad_i32: __self_0, cast: __self_1 } =>
                PassMode::Cast {
                    pad_i32: ::core::clone::Clone::clone(__self_0),
                    cast: ::core::clone::Clone::clone(__self_1),
                },
            PassMode::Indirect {
                attrs: __self_0, meta_attrs: __self_1, on_stack: __self_2 } =>
                PassMode::Indirect {
                    attrs: ::core::clone::Clone::clone(__self_0),
                    meta_attrs: ::core::clone::Clone::clone(__self_1),
                    on_stack: ::core::clone::Clone::clone(__self_2),
                },
        }
    }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for PassMode {
    #[inline]
    fn eq(&self, other: &PassMode) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr &&
            match (self, other) {
                (PassMode::Direct(__self_0), PassMode::Direct(__arg1_0)) =>
                    __self_0 == __arg1_0,
                (PassMode::Pair(__self_0, __self_1),
                    PassMode::Pair(__arg1_0, __arg1_1)) =>
                    __self_0 == __arg1_0 && __self_1 == __arg1_1,
                (PassMode::Cast { pad_i32: __self_0, cast: __self_1 },
                    PassMode::Cast { pad_i32: __arg1_0, cast: __arg1_1 }) =>
                    __self_0 == __arg1_0 && __self_1 == __arg1_1,
                (PassMode::Indirect {
                    attrs: __self_0, meta_attrs: __self_1, on_stack: __self_2 },
                    PassMode::Indirect {
                    attrs: __arg1_0, meta_attrs: __arg1_1, on_stack: __arg1_2 })
                    =>
                    __self_2 == __arg1_2 && __self_0 == __arg1_0 &&
                        __self_1 == __arg1_1,
                _ => true,
            }
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for PassMode {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<ArgAttributes>;
        let _: ::core::cmp::AssertParamIsEq<bool>;
        let _: ::core::cmp::AssertParamIsEq<Box<CastTarget>>;
        let _: ::core::cmp::AssertParamIsEq<Option<ArgAttributes>>;
    }
}Eq, #[automatically_derived]
impl ::core::hash::Hash for PassMode {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state);
        match self {
            PassMode::Direct(__self_0) =>
                ::core::hash::Hash::hash(__self_0, state),
            PassMode::Pair(__self_0, __self_1) => {
                ::core::hash::Hash::hash(__self_0, state);
                ::core::hash::Hash::hash(__self_1, state)
            }
            PassMode::Cast { pad_i32: __self_0, cast: __self_1 } => {
                ::core::hash::Hash::hash(__self_0, state);
                ::core::hash::Hash::hash(__self_1, state)
            }
            PassMode::Indirect {
                attrs: __self_0, meta_attrs: __self_1, on_stack: __self_2 } =>
                {
                ::core::hash::Hash::hash(__self_0, state);
                ::core::hash::Hash::hash(__self_1, state);
                ::core::hash::Hash::hash(__self_2, state)
            }
            _ => {}
        }
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for PassMode {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        match self {
            PassMode::Ignore =>
                ::core::fmt::Formatter::write_str(f, "Ignore"),
            PassMode::Direct(__self_0) =>
                ::core::fmt::Formatter::debug_tuple_field1_finish(f, "Direct",
                    &__self_0),
            PassMode::Pair(__self_0, __self_1) =>
                ::core::fmt::Formatter::debug_tuple_field2_finish(f, "Pair",
                    __self_0, &__self_1),
            PassMode::Cast { pad_i32: __self_0, cast: __self_1 } =>
                ::core::fmt::Formatter::debug_struct_field2_finish(f, "Cast",
                    "pad_i32", __self_0, "cast", &__self_1),
            PassMode::Indirect {
                attrs: __self_0, meta_attrs: __self_1, on_stack: __self_2 } =>
                ::core::fmt::Formatter::debug_struct_field3_finish(f,
                    "Indirect", "attrs", __self_0, "meta_attrs", __self_1,
                    "on_stack", &__self_2),
        }
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for PassMode {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                ::std::mem::discriminant(self).stable_hash(__hcx, __hasher);
                match *self {
                    PassMode::Ignore => {}
                    PassMode::Direct(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                    PassMode::Pair(ref __binding_0, ref __binding_1) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                    PassMode::Cast {
                        pad_i32: ref __binding_0, cast: ref __binding_1 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                    PassMode::Indirect {
                        attrs: ref __binding_0,
                        meta_attrs: ref __binding_1,
                        on_stack: ref __binding_2 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
40pub enum PassMode {
41    /// Ignore the argument.
42    ///
43    /// The argument is a ZST.
44    Ignore,
45    /// Pass the argument directly.
46    ///
47    /// The argument has a layout abi of `Scalar` or `Vector`.
48    /// Unfortunately due to past mistakes, in rare cases on wasm, it can also be `Aggregate`.
49    /// This is bad since it leaks LLVM implementation details into the ABI.
50    /// (Also see <https://github.com/rust-lang/rust/issues/115666>.)
51    Direct(ArgAttributes),
52    /// Pass a pair's elements directly in two arguments.
53    ///
54    /// The argument has a layout abi of `ScalarPair`.
55    Pair(ArgAttributes, ArgAttributes),
56    /// Pass the argument after casting it. See the `CastTarget` docs for details.
57    ///
58    /// `pad_i32` indicates if a `Reg::i32()` dummy argument is emitted before the real argument.
59    Cast { pad_i32: bool, cast: Box<CastTarget> },
60    /// Pass the argument indirectly via a hidden pointer.
61    ///
62    /// The `meta_attrs` value, if any, is for the metadata (vtable or length) of an unsized
63    /// argument. (This is the only mode that supports unsized arguments.)
64    ///
65    /// `on_stack` defines that the value should be passed at a fixed stack offset in accordance to
66    /// the ABI rather than passed using a pointer. This corresponds to the `byval` LLVM argument
67    /// attribute. The `byval` argument will use a byte array with the same size as the Rust type
68    /// (which ensures that padding is preserved and that we do not rely on LLVM's struct layout),
69    /// and will use the alignment specified in `attrs.pointee_align` (if `Some`) or the type's
70    /// alignment (if `None`). This means that the alignment will not always
71    /// match the Rust type's alignment; see documentation of `pass_by_stack_offset` for more info.
72    ///
73    /// `on_stack` cannot be true for unsized arguments, i.e., when `meta_attrs` is `Some`.
74    Indirect { attrs: ArgAttributes, meta_attrs: Option<ArgAttributes>, on_stack: bool },
75}
76
77impl PassMode {
78    /// Checks if these two `PassMode` are equal enough to be considered "the same for all
79    /// function call ABIs". However, the `Layout` can also impact ABI decisions,
80    /// so that needs to be compared as well!
81    pub fn eq_abi(&self, other: &Self) -> bool {
82        match (self, other) {
83            (PassMode::Ignore, PassMode::Ignore) => true,
84            (PassMode::Direct(a1), PassMode::Direct(a2)) => a1.eq_abi(a2),
85            (PassMode::Pair(a1, b1), PassMode::Pair(a2, b2)) => a1.eq_abi(a2) && b1.eq_abi(b2),
86            (
87                PassMode::Cast { cast: c1, pad_i32: pad1 },
88                PassMode::Cast { cast: c2, pad_i32: pad2 },
89            ) => c1.eq_abi(c2) && pad1 == pad2,
90            (
91                PassMode::Indirect { attrs: a1, meta_attrs: None, on_stack: s1 },
92                PassMode::Indirect { attrs: a2, meta_attrs: None, on_stack: s2 },
93            ) => a1.eq_abi(a2) && s1 == s2,
94            (
95                PassMode::Indirect { attrs: a1, meta_attrs: Some(e1), on_stack: s1 },
96                PassMode::Indirect { attrs: a2, meta_attrs: Some(e2), on_stack: s2 },
97            ) => a1.eq_abi(a2) && e1.eq_abi(e2) && s1 == s2,
98            _ => false,
99        }
100    }
101}
102
103// Hack to disable non_upper_case_globals only for the bitflags! and not for the rest
104// of this module
105pub use attr_impl::ArgAttribute;
106
107#[allow(non_upper_case_globals)]
108#[allow(unused)]
109mod attr_impl {
110    use rustc_macros::StableHash;
111
112    // The subset of llvm::Attribute needed for arguments, packed into a bitfield.
113    #[derive(#[automatically_derived]
impl ::core::clone::Clone for ArgAttribute {
    #[inline]
    fn clone(&self) -> ArgAttribute {
        let _: ::core::clone::AssertParamIsClone<u16>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for ArgAttribute { }Copy, #[automatically_derived]
impl ::core::default::Default for ArgAttribute {
    #[inline]
    fn default() -> ArgAttribute {
        ArgAttribute(::core::default::Default::default())
    }
}Default, #[automatically_derived]
impl ::core::hash::Hash for ArgAttribute {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.0, state)
    }
}Hash, #[automatically_derived]
impl ::core::cmp::PartialEq for ArgAttribute {
    #[inline]
    fn eq(&self, other: &ArgAttribute) -> bool { self.0 == other.0 }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for ArgAttribute {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<u16>;
    }
}Eq, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for ArgAttribute
            {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    ArgAttribute(ref __binding_0) => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
114    pub struct ArgAttribute(u16);
115    impl ArgAttribute {
    #[allow(deprecated, non_upper_case_globals,)]
    pub const CapturesNone: Self = Self::from_bits_retain(0b111);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const CapturesAddress: Self = Self::from_bits_retain(0b110);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const CapturesReadOnly: Self = Self::from_bits_retain(0b100);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const NoAlias: Self = Self::from_bits_retain(1 << 3);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const NonNull: Self = Self::from_bits_retain(1 << 4);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const ReadOnly: Self = Self::from_bits_retain(1 << 5);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const InReg: Self = Self::from_bits_retain(1 << 6);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const NoUndef: Self = Self::from_bits_retain(1 << 7);
    #[allow(deprecated, non_upper_case_globals,)]
    pub const Writable: Self = Self::from_bits_retain(1 << 8);
    #[doc =
    r" It is UB for this pointer or any pointer derived from it to be used for"]
    #[doc =
    r" deallocation (except for zero-sized deallocation) while the function is"]
    #[doc =
    r" executing. Only valid on arguments (including return values that are passed"]
    #[doc = r" indirectly as arguments)."]
    #[allow(deprecated, non_upper_case_globals,)]
    pub const NoFree: Self = Self::from_bits_retain(1 << 9);
}
impl ::bitflags::Flags for ArgAttribute {
    const FLAGS: &'static [::bitflags::Flag<ArgAttribute>] =
        &[{

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

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

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

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

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

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

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

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

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

                        #[allow(deprecated, non_upper_case_globals,)]
                        ::bitflags::Flag::new("NoFree", ArgAttribute::NoFree)
                    }];
    type Bits = u16;
    fn bits(&self) -> u16 { ArgAttribute::bits(self) }
    fn from_bits_retain(bits: u16) -> ArgAttribute {
        ArgAttribute::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 :: iter_without_into_iter,)]
const _: () =
    {
        #[allow(dead_code, deprecated, unused_attributes)]
        impl ArgAttribute {
            /// Get a flags value with all bits unset.
            #[inline]
            pub const fn empty() -> Self {
                Self(<u16 as ::bitflags::Bits>::EMPTY)
            }
            /// Get a flags value with all known bits set.
            #[inline]
            pub const fn all() -> Self {
                let mut truncated = <u16 as ::bitflags::Bits>::EMPTY;
                let mut i = 0;
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute as
                                            ::bitflags::Flags>::FLAGS[i].value().bits();
                        truncated = truncated | flag;
                        i += 1;
                    }
                };
                {
                    {
                        let flag =
                            <ArgAttribute 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) -> u16 { 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: u16)
                -> ::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: u16) -> Self {
                Self(bits & Self::all().0)
            }
            /// Convert from a bits value exactly.
            #[inline]
            pub const fn from_bits_retain(bits: u16) -> 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 == "CapturesNone" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::CapturesNone.bits()));
                    }
                };
                ;
                {
                    if name == "CapturesAddress" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::CapturesAddress.bits()));
                    }
                };
                ;
                {
                    if name == "CapturesReadOnly" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::CapturesReadOnly.bits()));
                    }
                };
                ;
                {
                    if name == "NoAlias" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::NoAlias.bits()));
                    }
                };
                ;
                {
                    if name == "NonNull" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::NonNull.bits()));
                    }
                };
                ;
                {
                    if name == "ReadOnly" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::ReadOnly.bits()));
                    }
                };
                ;
                {
                    if name == "InReg" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::InReg.bits()));
                    }
                };
                ;
                {
                    if name == "NoUndef" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::NoUndef.bits()));
                    }
                };
                ;
                {
                    if name == "Writable" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::Writable.bits()));
                    }
                };
                ;
                {
                    if name == "NoFree" {
                        return ::bitflags::__private::core::option::Option::Some(Self(ArgAttribute::NoFree.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 == <u16 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 != <u16 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 ArgAttribute {
            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 ArgAttribute {
            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 ArgAttribute {
            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 ArgAttribute {
            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 ArgAttribute {
            type Output = Self;
            /// The bitwise or (`|`) of the bits in two flags values.
            #[inline]
            fn bitor(self, other: ArgAttribute) -> Self { self.union(other) }
        }
        impl ::bitflags::__private::core::ops::BitOrAssign for ArgAttribute {
            /// 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 ArgAttribute {
            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 ArgAttribute {
            /// 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 ArgAttribute {
            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 ArgAttribute {
            /// 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 ArgAttribute {
            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 ArgAttribute {
            /// 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 ArgAttribute {
            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<ArgAttribute> for
            ArgAttribute {
            /// 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<ArgAttribute> for
            ArgAttribute {
            /// 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 ArgAttribute {
            /// 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<ArgAttribute> {
                ::bitflags::iter::Iter::__private_const_new(<ArgAttribute as
                        ::bitflags::Flags>::FLAGS,
                    ArgAttribute::from_bits_retain(self.bits()),
                    ArgAttribute::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<ArgAttribute> {
                ::bitflags::iter::IterNames::__private_const_new(<ArgAttribute
                        as ::bitflags::Flags>::FLAGS,
                    ArgAttribute::from_bits_retain(self.bits()),
                    ArgAttribute::from_bits_retain(self.bits()))
            }
        }
        impl ::bitflags::__private::core::iter::IntoIterator for ArgAttribute
            {
            type Item = ArgAttribute;
            type IntoIter = ::bitflags::iter::Iter<ArgAttribute>;
            fn into_iter(self) -> Self::IntoIter { self.iter() }
        }
    };bitflags::bitflags! {
116        impl ArgAttribute: u16 {
117            const CapturesNone     = 0b111;
118            const CapturesAddress  = 0b110;
119            const CapturesReadOnly = 0b100;
120            const NoAlias  = 1 << 3;
121            const NonNull  = 1 << 4;
122            const ReadOnly = 1 << 5;
123            const InReg    = 1 << 6;
124            const NoUndef  = 1 << 7;
125            const Writable = 1 << 8;
126            /// It is UB for this pointer or any pointer derived from it to be used for
127            /// deallocation (except for zero-sized deallocation) while the function is
128            /// executing. Only valid on arguments (including return values that are passed
129            /// indirectly as arguments).
130            const NoFree   = 1 << 9;
131        }
132    }
133    impl ::std::fmt::Debug for ArgAttribute {
    fn fmt(&self, f: &mut ::std::fmt::Formatter<'_>) -> ::std::fmt::Result {
        ::bitflags::parser::to_writer(self, f)
    }
}rustc_data_structures::external_bitflags_debug! { ArgAttribute }
134}
135
136/// Sometimes an ABI requires small integers to be extended to a full or partial register. This enum
137/// defines if this extension should be zero-extension or sign-extension when necessary. When it is
138/// not necessary to extend the argument, this enum is ignored.
139#[derive(#[automatically_derived]
impl ::core::marker::Copy for ArgExtension { }Copy, #[automatically_derived]
impl ::core::clone::Clone for ArgExtension {
    #[inline]
    fn clone(&self) -> ArgExtension { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for ArgExtension {
    #[inline]
    fn eq(&self, other: &ArgExtension) -> 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 ArgExtension {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq, #[automatically_derived]
impl ::core::hash::Hash for ArgExtension {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state)
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for ArgExtension {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                ArgExtension::None => "None",
                ArgExtension::Zext => "Zext",
                ArgExtension::Sext => "Sext",
            })
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for ArgExtension
            {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                ::std::mem::discriminant(self).stable_hash(__hcx, __hasher);
                match *self {
                    ArgExtension::None => {}
                    ArgExtension::Zext => {}
                    ArgExtension::Sext => {}
                }
            }
        }
    };StableHash)]
140pub enum ArgExtension {
141    None,
142    Zext,
143    Sext,
144}
145
146/// A compact representation of LLVM attributes (at least those relevant for this module)
147/// that can be manipulated without interacting with LLVM's Attribute machinery.
148#[derive(#[automatically_derived]
impl ::core::marker::Copy for ArgAttributes { }Copy, #[automatically_derived]
impl ::core::clone::Clone for ArgAttributes {
    #[inline]
    fn clone(&self) -> ArgAttributes {
        let _: ::core::clone::AssertParamIsClone<ArgAttribute>;
        let _: ::core::clone::AssertParamIsClone<ArgExtension>;
        let _: ::core::clone::AssertParamIsClone<Size>;
        let _: ::core::clone::AssertParamIsClone<Option<Align>>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for ArgAttributes {
    #[inline]
    fn eq(&self, other: &ArgAttributes) -> bool {
        self.regular == other.regular && self.arg_ext == other.arg_ext &&
                self.pointee_size == other.pointee_size &&
            self.pointee_align == other.pointee_align
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for ArgAttributes {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<ArgAttribute>;
        let _: ::core::cmp::AssertParamIsEq<ArgExtension>;
        let _: ::core::cmp::AssertParamIsEq<Size>;
        let _: ::core::cmp::AssertParamIsEq<Option<Align>>;
    }
}Eq, #[automatically_derived]
impl ::core::hash::Hash for ArgAttributes {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.regular, state);
        ::core::hash::Hash::hash(&self.arg_ext, state);
        ::core::hash::Hash::hash(&self.pointee_size, state);
        ::core::hash::Hash::hash(&self.pointee_align, state)
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for ArgAttributes {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field4_finish(f, "ArgAttributes",
            "regular", &self.regular, "arg_ext", &self.arg_ext,
            "pointee_size", &self.pointee_size, "pointee_align",
            &&self.pointee_align)
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for
            ArgAttributes {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    ArgAttributes {
                        regular: ref __binding_0,
                        arg_ext: ref __binding_1,
                        pointee_size: ref __binding_2,
                        pointee_align: ref __binding_3 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                        { __binding_3.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
149pub struct ArgAttributes {
150    pub regular: ArgAttribute,
151    pub arg_ext: ArgExtension,
152    /// If the pointer is not null, the minimum dereferenceable size of the pointee, at the time of
153    /// function entry (for arguments) or function return (for return values).
154    pub pointee_size: Size,
155    /// The minimum alignment of the pointee, if any.
156    pub pointee_align: Option<Align>,
157}
158
159impl ArgAttributes {
160    pub fn new() -> Self {
161        ArgAttributes {
162            regular: ArgAttribute::default(),
163            arg_ext: ArgExtension::None,
164            pointee_size: Size::ZERO,
165            pointee_align: None,
166        }
167    }
168
169    pub fn ext(&mut self, ext: ArgExtension) -> &mut Self {
170        if !(self.arg_ext == ArgExtension::None || self.arg_ext == ext) {
    {
        ::core::panicking::panic_fmt(format_args!("cannot set {0:?} when {1:?} is already set",
                ext, self.arg_ext));
    }
};assert!(
171            self.arg_ext == ArgExtension::None || self.arg_ext == ext,
172            "cannot set {:?} when {:?} is already set",
173            ext,
174            self.arg_ext
175        );
176        self.arg_ext = ext;
177        self
178    }
179
180    pub fn set(&mut self, attr: ArgAttribute) -> &mut Self {
181        self.regular |= attr;
182        self
183    }
184
185    pub fn contains(&self, attr: ArgAttribute) -> bool {
186        self.regular.contains(attr)
187    }
188
189    /// Checks if these two `ArgAttributes` are equal enough to be considered "the same for all
190    /// function call ABIs".
191    pub fn eq_abi(&self, other: &Self) -> bool {
192        // There's only one regular attribute that matters for the call ABI: InReg.
193        // Everything else is things like noalias, dereferenceable, nonnull, ...
194        // (This also applies to pointee_size, pointee_align.)
195        if self.regular.contains(ArgAttribute::InReg) != other.regular.contains(ArgAttribute::InReg)
196        {
197            return false;
198        }
199        // We also compare the sign extension mode -- this could let the callee make assumptions
200        // about bits that conceptually were not even passed.
201        if self.arg_ext != other.arg_ext {
202            return false;
203        }
204        true
205    }
206}
207
208impl From<ArgAttribute> for ArgAttributes {
209    fn from(value: ArgAttribute) -> Self {
210        Self {
211            regular: value,
212            arg_ext: ArgExtension::None,
213            pointee_size: Size::ZERO,
214            pointee_align: None,
215        }
216    }
217}
218
219/// An argument passed entirely registers with the
220/// same kind (e.g., HFA / HVA on PPC64 and AArch64).
221#[derive(#[automatically_derived]
impl ::core::clone::Clone for Uniform {
    #[inline]
    fn clone(&self) -> Uniform {
        let _: ::core::clone::AssertParamIsClone<Reg>;
        let _: ::core::clone::AssertParamIsClone<Size>;
        let _: ::core::clone::AssertParamIsClone<bool>;
        *self
    }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for Uniform { }Copy, #[automatically_derived]
impl ::core::cmp::PartialEq for Uniform {
    #[inline]
    fn eq(&self, other: &Uniform) -> bool {
        self.is_consecutive == other.is_consecutive && self.unit == other.unit
            && self.total == other.total
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for Uniform {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<Reg>;
        let _: ::core::cmp::AssertParamIsEq<Size>;
        let _: ::core::cmp::AssertParamIsEq<bool>;
    }
}Eq, #[automatically_derived]
impl ::core::hash::Hash for Uniform {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.unit, state);
        ::core::hash::Hash::hash(&self.total, state);
        ::core::hash::Hash::hash(&self.is_consecutive, state)
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for Uniform {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field3_finish(f, "Uniform",
            "unit", &self.unit, "total", &self.total, "is_consecutive",
            &&self.is_consecutive)
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for Uniform {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    Uniform {
                        unit: ref __binding_0,
                        total: ref __binding_1,
                        is_consecutive: ref __binding_2 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
222pub struct Uniform {
223    pub unit: Reg,
224
225    /// The total size of the argument, which can be:
226    /// * equal to `unit.size` (one scalar/vector),
227    /// * a multiple of `unit.size` (an array of scalar/vectors),
228    /// * if `unit.kind` is `Integer`, the last element can be shorter, i.e., `{ i64, i64, i32 }`
229    ///   for 64-bit integers with a total size of 20 bytes. When the argument is actually passed,
230    ///   this size will be rounded up to the nearest multiple of `unit.size`.
231    pub total: Size,
232
233    /// Indicate that the argument is consecutive, in the sense that either all values need to be
234    /// passed in register, or all on the stack. If they are passed on the stack, there should be
235    /// no additional padding between elements.
236    pub is_consecutive: bool,
237}
238
239impl From<Reg> for Uniform {
240    fn from(unit: Reg) -> Uniform {
241        Uniform { unit, total: unit.size, is_consecutive: false }
242    }
243}
244
245impl Uniform {
246    pub fn align<C: HasDataLayout>(&self, cx: &C) -> Align {
247        self.unit.align(cx)
248    }
249
250    /// Pass using one or more values of the given type, without requiring them to be consecutive.
251    /// That is, some values may be passed in register and some on the stack.
252    pub fn new(unit: Reg, total: Size) -> Self {
253        Uniform { unit, total, is_consecutive: false }
254    }
255
256    /// Pass using one or more consecutive values of the given type. Either all values will be
257    /// passed in registers, or all on the stack.
258    pub fn consecutive(unit: Reg, total: Size) -> Self {
259        Uniform { unit, total, is_consecutive: true }
260    }
261}
262
263/// Describes the type used for `PassMode::Cast`.
264///
265/// Passing arguments in this mode works as follows: the registers in the `prefix` (the ones that
266/// are `Some`) get laid out one after the other (using `repr(C)` layout rules). Then the
267/// `rest.unit` register type gets repeated often enough to cover `rest.size`. This describes the
268/// actual type used for the call; the Rust type of the argument is then transmuted to this ABI type
269/// (and all data in the padding between the registers is dropped).
270#[derive(#[automatically_derived]
impl ::core::clone::Clone for CastTarget {
    #[inline]
    fn clone(&self) -> CastTarget {
        CastTarget {
            prefix: ::core::clone::Clone::clone(&self.prefix),
            rest_offset: ::core::clone::Clone::clone(&self.rest_offset),
            rest: ::core::clone::Clone::clone(&self.rest),
            attrs: ::core::clone::Clone::clone(&self.attrs),
        }
    }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for CastTarget {
    #[inline]
    fn eq(&self, other: &CastTarget) -> bool {
        self.prefix == other.prefix && self.rest_offset == other.rest_offset
                && self.rest == other.rest && self.attrs == other.attrs
    }
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for CastTarget {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<ArrayVec<Reg, 8>>;
        let _: ::core::cmp::AssertParamIsEq<Option<Size>>;
        let _: ::core::cmp::AssertParamIsEq<Uniform>;
        let _: ::core::cmp::AssertParamIsEq<ArgAttributes>;
    }
}Eq, #[automatically_derived]
impl ::core::hash::Hash for CastTarget {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.prefix, state);
        ::core::hash::Hash::hash(&self.rest_offset, state);
        ::core::hash::Hash::hash(&self.rest, state);
        ::core::hash::Hash::hash(&self.attrs, state)
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for CastTarget {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field4_finish(f, "CastTarget",
            "prefix", &self.prefix, "rest_offset", &self.rest_offset, "rest",
            &self.rest, "attrs", &&self.attrs)
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for CastTarget {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    CastTarget {
                        prefix: ref __binding_0,
                        rest_offset: ref __binding_1,
                        rest: ref __binding_2,
                        attrs: ref __binding_3 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                        { __binding_3.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
271pub struct CastTarget {
272    // Note that this is fixed to 8 elements for now as ABIs currently don't
273    // need anything further beyond that, and when this code was originally
274    // refactored to use `ArrayVec` it was already using 8, so that stuck
275    // around.
276    pub prefix: ArrayVec<Reg, 8>,
277    /// The offset of `rest` from the start of the value. Currently only implemented for a `Reg`
278    /// pair created by the `offset_pair` method.
279    pub rest_offset: Option<Size>,
280    pub rest: Uniform,
281    pub attrs: ArgAttributes,
282}
283
284impl From<Reg> for CastTarget {
285    fn from(unit: Reg) -> CastTarget {
286        CastTarget::from(Uniform::from(unit))
287    }
288}
289
290impl From<Uniform> for CastTarget {
291    fn from(uniform: Uniform) -> CastTarget {
292        Self::prefixed(Default::default(), uniform)
293    }
294}
295
296impl CastTarget {
297    pub fn prefixed(prefix: ArrayVec<Reg, 8>, rest: Uniform) -> Self {
298        Self { prefix, rest_offset: None, rest, attrs: ArgAttributes::new() }
299    }
300
301    pub fn offset_pair(a: Reg, offset_from_start: Size, b: Reg) -> Self {
302        let mut prefix = ArrayVec::new();
303        prefix.push(a);
304        Self {
305            prefix,
306            rest_offset: Some(offset_from_start),
307            rest: b.into(),
308            attrs: ArgAttributes::new(),
309        }
310    }
311
312    pub fn with_attrs(mut self, attrs: ArgAttributes) -> Self {
313        self.attrs = attrs;
314        self
315    }
316
317    pub fn pair(a: Reg, b: Reg) -> CastTarget {
318        let mut prefix = ArrayVec::new();
319        prefix.push(a);
320        Self::prefixed(prefix, Uniform::from(b))
321    }
322
323    /// When you only access the range containing valid data, you can use this unaligned size;
324    /// otherwise, use the safer `size` method.
325    pub fn unaligned_size<C: HasDataLayout>(&self, _cx: &C) -> Size {
326        // Prefix arguments are passed in specific designated registers
327        let prefix_size = if let Some(offset_from_start) = self.rest_offset {
328            offset_from_start
329        } else {
330            self.prefix.iter().map(|reg| reg.size).fold(Size::ZERO, |acc, size| acc + size)
331        };
332        // Remaining arguments are passed in chunks of the unit size
333        let rest_size =
334            self.rest.unit.size * self.rest.total.bytes().div_ceil(self.rest.unit.size.bytes());
335
336        prefix_size + rest_size
337    }
338
339    pub fn size<C: HasDataLayout>(&self, cx: &C) -> Size {
340        self.unaligned_size(cx).align_to(self.align(cx))
341    }
342
343    pub fn align<C: HasDataLayout>(&self, cx: &C) -> Align {
344        self.prefix
345            .iter()
346            .map(|reg| reg.align(cx))
347            .fold(cx.data_layout().aggregate_align.max(self.rest.align(cx)), |acc, align| {
348                acc.max(align)
349            })
350    }
351
352    /// Checks if these two `CastTarget` are equal enough to be considered "the same for all
353    /// function call ABIs".
354    pub fn eq_abi(&self, other: &Self) -> bool {
355        let CastTarget {
356            prefix: prefix_l,
357            rest_offset: rest_offset_l,
358            rest: rest_l,
359            attrs: attrs_l,
360        } = self;
361        let CastTarget {
362            prefix: prefix_r,
363            rest_offset: rest_offset_r,
364            rest: rest_r,
365            attrs: attrs_r,
366        } = other;
367        prefix_l == prefix_r
368            && rest_offset_l == rest_offset_r
369            && rest_l == rest_r
370            && attrs_l.eq_abi(attrs_r)
371    }
372}
373
374/// Information about how to pass an argument to,
375/// or return a value from, a function, under some ABI.
376#[derive(#[automatically_derived]
impl<'a, Ty: ::core::clone::Clone> ::core::clone::Clone for ArgAbi<'a, Ty> {
    #[inline]
    fn clone(&self) -> ArgAbi<'a, Ty> {
        ArgAbi {
            layout: ::core::clone::Clone::clone(&self.layout),
            mode: ::core::clone::Clone::clone(&self.mode),
        }
    }
}Clone, #[automatically_derived]
impl<'a, Ty: ::core::cmp::PartialEq> ::core::cmp::PartialEq for ArgAbi<'a, Ty>
    {
    #[inline]
    fn eq(&self, other: &ArgAbi<'a, Ty>) -> bool {
        self.layout == other.layout && self.mode == other.mode
    }
}PartialEq, #[automatically_derived]
impl<'a, Ty: ::core::cmp::Eq> ::core::cmp::Eq for ArgAbi<'a, Ty> {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<TyAndLayout<'a, Ty>>;
        let _: ::core::cmp::AssertParamIsEq<PassMode>;
    }
}Eq, #[automatically_derived]
impl<'a, Ty: ::core::hash::Hash> ::core::hash::Hash for ArgAbi<'a, Ty> {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.layout, state);
        ::core::hash::Hash::hash(&self.mode, state)
    }
}Hash, const _: () =
    {
        impl<'a, Ty> ::rustc_data_structures::stable_hash::StableHash for
            ArgAbi<'a, Ty> where
            Ty: ::rustc_data_structures::stable_hash::StableHash {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    ArgAbi { layout: ref __binding_0, mode: ref __binding_1 } =>
                        {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
377pub struct ArgAbi<'a, Ty> {
378    pub layout: TyAndLayout<'a, Ty>,
379    pub mode: PassMode,
380}
381
382// Needs to be a custom impl because of the bounds on the `TyAndLayout` debug impl.
383impl<'a, Ty: fmt::Display> fmt::Debug for ArgAbi<'a, Ty> {
384    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
385        let ArgAbi { layout, mode } = self;
386        f.debug_struct("ArgAbi").field("layout", layout).field("mode", mode).finish()
387    }
388}
389
390impl<'a, Ty> ArgAbi<'a, Ty> {
391    /// This defines the "default ABI" for that type, that is then later adjusted in `fn_abi_adjust_for_abi`.
392    pub fn new(
393        cx: &impl HasDataLayout,
394        layout: TyAndLayout<'a, Ty>,
395        scalar_attrs: impl Fn(Scalar, Size) -> ArgAttributes,
396    ) -> Self {
397        let mode = match layout.backend_repr {
398            _ if layout.is_zst() => PassMode::Ignore,
399            BackendRepr::Scalar(scalar) => PassMode::Direct(scalar_attrs(scalar, Size::ZERO)),
400            BackendRepr::ScalarPair(a, b) => PassMode::Pair(
401                scalar_attrs(a, Size::ZERO),
402                scalar_attrs(b, a.size(cx).align_to(b.align(cx).abi)),
403            ),
404            BackendRepr::SimdVector { .. } => PassMode::Direct(ArgAttributes::new()),
405            BackendRepr::Memory { .. } => Self::indirect_pass_mode(&layout),
406            BackendRepr::SimdScalableVector { .. } => PassMode::Direct(ArgAttributes::new()),
407        };
408        ArgAbi { layout, mode }
409    }
410
411    fn indirect_pass_mode(layout: &TyAndLayout<'a, Ty>) -> PassMode {
412        let mut attrs = ArgAttributes::new();
413
414        // For non-immediate arguments the callee gets its own copy of
415        // the value on the stack, so there are no aliases. The function
416        // can capture the address of the argument, but not the provenance.
417        attrs
418            .set(ArgAttribute::NoAlias)
419            .set(ArgAttribute::CapturesAddress)
420            .set(ArgAttribute::NonNull)
421            .set(ArgAttribute::NoUndef)
422            .set(ArgAttribute::NoFree);
423        attrs.pointee_size = layout.size;
424        attrs.pointee_align = Some(layout.align.abi);
425
426        let meta_attrs = layout.is_unsized().then_some(ArgAttributes::new());
427
428        PassMode::Indirect { attrs, meta_attrs, on_stack: false }
429    }
430
431    /// Pass this argument directly instead. Should NOT be used!
432    /// Only exists because of past ABI mistakes that will take time to fix
433    /// (see <https://github.com/rust-lang/rust/issues/115666>).
434    #[track_caller]
435    pub fn make_direct_deprecated(&mut self) {
436        match self.mode {
437            PassMode::Indirect { .. } => {
438                self.mode = PassMode::Direct(ArgAttributes::new());
439            }
440            PassMode::Ignore | PassMode::Direct(_) | PassMode::Pair(_, _) => {} // already direct
441            _ => {
    ::core::panicking::panic_fmt(format_args!("Tried to make {0:?} direct",
            self.mode));
}panic!("Tried to make {:?} direct", self.mode),
442        }
443    }
444
445    /// Pass this argument indirectly, by passing a (thin or wide) pointer to the argument instead.
446    /// This is valid for both sized and unsized arguments.
447    #[track_caller]
448    pub fn make_indirect(&mut self) {
449        match self.mode {
450            PassMode::Direct(_) | PassMode::Pair(_, _) => {
451                self.mode = Self::indirect_pass_mode(&self.layout);
452            }
453            PassMode::Indirect { attrs: _, meta_attrs: _, on_stack: false } => {
454                // already indirect
455            }
456            _ => {
    ::core::panicking::panic_fmt(format_args!("Tried to make {0:?} indirect",
            self.mode));
}panic!("Tried to make {:?} indirect", self.mode),
457        }
458    }
459
460    /// Same as `make_indirect`, but for arguments that are ignored. Only needed for ABIs that pass
461    /// ZSTs indirectly.
462    #[track_caller]
463    pub fn make_indirect_from_ignore(&mut self) {
464        match self.mode {
465            PassMode::Ignore => {
466                self.mode = Self::indirect_pass_mode(&self.layout);
467            }
468            PassMode::Indirect { attrs: _, meta_attrs: _, on_stack: false } => {
469                // already indirect
470            }
471            _ => {
    ::core::panicking::panic_fmt(format_args!("Tried to make {0:?} indirect (expected `PassMode::Ignore`)",
            self.mode));
}panic!("Tried to make {:?} indirect (expected `PassMode::Ignore`)", self.mode),
472        }
473    }
474
475    /// Pass this argument indirectly, by placing it at a fixed stack offset.
476    /// This corresponds to the `byval` LLVM argument attribute.
477    /// This is only valid for sized arguments.
478    ///
479    /// `byval_align` specifies the alignment of the `byval` stack slot, which does not need to
480    /// correspond to the type's alignment. This will be `Some` if the target's ABI specifies that
481    /// stack slots used for arguments passed by-value have specific alignment requirements which
482    /// differ from the alignment used in other situations.
483    ///
484    /// If `None`, the type's alignment is used.
485    ///
486    /// If the resulting alignment differs from the type's alignment,
487    /// the argument will be copied to an alloca with sufficient alignment,
488    /// either in the caller (if the type's alignment is lower than the byval alignment)
489    /// or in the callee (if the type's alignment is higher than the byval alignment),
490    /// to ensure that Rust code never sees an underaligned pointer.
491    pub fn pass_by_stack_offset(&mut self, byval_align: Option<Align>) {
492        if !!self.layout.is_unsized() {
    {
        ::core::panicking::panic_fmt(format_args!("used byval ABI for unsized layout"));
    }
};assert!(!self.layout.is_unsized(), "used byval ABI for unsized layout");
493        self.make_indirect();
494        match self.mode {
495            PassMode::Indirect { ref mut attrs, meta_attrs: _, ref mut on_stack } => {
496                *on_stack = true;
497
498                // Some platforms, like 32-bit x86, change the alignment of the type when passing
499                // `byval`. Account for that.
500                if let Some(byval_align) = byval_align {
501                    // On all targets with byval align this is currently true, so let's assert it.
502                    if true {
    if !(byval_align >= Align::from_bytes(4).unwrap()) {
        ::core::panicking::panic("assertion failed: byval_align >= Align::from_bytes(4).unwrap()")
    };
};debug_assert!(byval_align >= Align::from_bytes(4).unwrap());
503                    attrs.pointee_align = Some(byval_align);
504                }
505            }
506            _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
507        }
508    }
509
510    pub fn extend_integer_width_to(&mut self, bits: u64) {
511        // Only integers have signedness
512        if let BackendRepr::Scalar(scalar) = self.layout.backend_repr
513            && let Primitive::Int(i, signed) = scalar.primitive()
514            && i.size().bits() < bits
515            && let PassMode::Direct(ref mut attrs) = self.mode
516        {
517            if signed {
518                attrs.ext(ArgExtension::Sext)
519            } else {
520                attrs.ext(ArgExtension::Zext)
521            };
522        }
523    }
524
525    pub fn cast_to<T: Into<CastTarget>>(&mut self, target: T) {
526        self.mode = PassMode::Cast { cast: Box::new(target.into()), pad_i32: false };
527    }
528
529    pub fn cast_to_with_attrs<T: Into<CastTarget>>(&mut self, target: T, attrs: ArgAttributes) {
530        self.mode =
531            PassMode::Cast { cast: Box::new(target.into().with_attrs(attrs)), pad_i32: false };
532    }
533
534    pub fn cast_to_and_pad_i32<T: Into<CastTarget>>(&mut self, target: T, pad_i32: bool) {
535        self.mode = PassMode::Cast { cast: Box::new(target.into()), pad_i32 };
536    }
537
538    pub fn is_indirect(&self) -> bool {
539        #[allow(non_exhaustive_omitted_patterns)] match self.mode {
    PassMode::Indirect { .. } => true,
    _ => false,
}matches!(self.mode, PassMode::Indirect { .. })
540    }
541
542    pub fn is_sized_indirect(&self) -> bool {
543        #[allow(non_exhaustive_omitted_patterns)] match self.mode {
    PassMode::Indirect { attrs: _, meta_attrs: None, on_stack: _ } => true,
    _ => false,
}matches!(self.mode, PassMode::Indirect { attrs: _, meta_attrs: None, on_stack: _ })
544    }
545
546    pub fn is_unsized_indirect(&self) -> bool {
547        #[allow(non_exhaustive_omitted_patterns)] match self.mode {
    PassMode::Indirect { attrs: _, meta_attrs: Some(_), on_stack: _ } => true,
    _ => false,
}matches!(self.mode, PassMode::Indirect { attrs: _, meta_attrs: Some(_), on_stack: _ })
548    }
549
550    pub fn is_ignore(&self) -> bool {
551        #[allow(non_exhaustive_omitted_patterns)] match self.mode {
    PassMode::Ignore => true,
    _ => false,
}matches!(self.mode, PassMode::Ignore)
552    }
553
554    /// Checks if these two `ArgAbi` are equal enough to be considered "the same for all
555    /// function call ABIs".
556    pub fn eq_abi(&self, other: &Self) -> bool
557    where
558        Ty: PartialEq,
559    {
560        // Ideally we'd just compare the `mode`, but that is not enough -- for some modes LLVM will look
561        // at the type.
562        self.layout.eq_abi(&other.layout) && self.mode.eq_abi(&other.mode) && {
563            // `fn_arg_sanity_check` accepts `PassMode::Direct` for some aggregates.
564            // That elevates any type difference to an ABI difference since we just use the
565            // full Rust type as the LLVM argument/return type.
566            if #[allow(non_exhaustive_omitted_patterns)] match self.mode {
    PassMode::Direct(..) => true,
    _ => false,
}matches!(self.mode, PassMode::Direct(..))
567                && #[allow(non_exhaustive_omitted_patterns)] match self.layout.backend_repr {
    BackendRepr::Memory { .. } => true,
    _ => false,
}matches!(self.layout.backend_repr, BackendRepr::Memory { .. })
568            {
569                // For aggregates in `Direct` mode to be compatible, the types need to be equal.
570                self.layout.ty == other.layout.ty
571            } else {
572                true
573            }
574        }
575    }
576}
577
578#[derive(#[automatically_derived]
impl ::core::marker::Copy for RiscvInterruptKind { }Copy, #[automatically_derived]
impl ::core::clone::Clone for RiscvInterruptKind {
    #[inline]
    fn clone(&self) -> RiscvInterruptKind { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for RiscvInterruptKind {
    #[inline]
    fn eq(&self, other: &RiscvInterruptKind) -> 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 RiscvInterruptKind {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq, #[automatically_derived]
impl ::core::hash::Hash for RiscvInterruptKind {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        ::core::hash::Hash::hash(&__self_discr, state)
    }
}Hash, #[automatically_derived]
impl ::core::fmt::Debug for RiscvInterruptKind {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                RiscvInterruptKind::Machine => "Machine",
                RiscvInterruptKind::Supervisor => "Supervisor",
            })
    }
}Debug, const _: () =
    {
        impl ::rustc_data_structures::stable_hash::StableHash for
            RiscvInterruptKind {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                ::std::mem::discriminant(self).stable_hash(__hcx, __hasher);
                match *self {
                    RiscvInterruptKind::Machine => {}
                    RiscvInterruptKind::Supervisor => {}
                }
            }
        }
    };StableHash)]
579pub enum RiscvInterruptKind {
580    Machine,
581    Supervisor,
582}
583
584impl RiscvInterruptKind {
585    pub fn as_str(&self) -> &'static str {
586        match self {
587            Self::Machine => "machine",
588            Self::Supervisor => "supervisor",
589        }
590    }
591}
592
593/// Metadata describing how the arguments to a native function
594/// should be passed in order to respect the native ABI.
595///
596/// The signature represented by this type may not match the MIR function signature.
597/// Certain attributes, like `#[track_caller]` can introduce additional arguments, which are present in [`FnAbi`], but not in `FnSig`.
598/// The std::offload module also adds an addition dyn_ptr argument to the GpuKernel ABI.
599/// While this difference is rarely relevant, it should still be kept in mind.
600///
601/// I will do my best to describe this structure, but these
602/// comments are reverse-engineered and may be inaccurate. -NDM
603#[derive(#[automatically_derived]
impl<'a, Ty: ::core::clone::Clone> ::core::clone::Clone for FnAbi<'a, Ty> {
    #[inline]
    fn clone(&self) -> FnAbi<'a, Ty> {
        FnAbi {
            args: ::core::clone::Clone::clone(&self.args),
            ret: ::core::clone::Clone::clone(&self.ret),
            c_variadic: ::core::clone::Clone::clone(&self.c_variadic),
            fixed_count: ::core::clone::Clone::clone(&self.fixed_count),
            conv: ::core::clone::Clone::clone(&self.conv),
            can_unwind: ::core::clone::Clone::clone(&self.can_unwind),
        }
    }
}Clone, #[automatically_derived]
impl<'a, Ty: ::core::cmp::PartialEq> ::core::cmp::PartialEq for FnAbi<'a, Ty>
    {
    #[inline]
    fn eq(&self, other: &FnAbi<'a, Ty>) -> bool {
        self.c_variadic == other.c_variadic &&
                            self.fixed_count == other.fixed_count &&
                        self.can_unwind == other.can_unwind &&
                    self.args == other.args && self.ret == other.ret &&
            self.conv == other.conv
    }
}PartialEq, #[automatically_derived]
impl<'a, Ty: ::core::cmp::Eq> ::core::cmp::Eq for FnAbi<'a, Ty> {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {
        let _: ::core::cmp::AssertParamIsEq<Box<[ArgAbi<'a, Ty>]>>;
        let _: ::core::cmp::AssertParamIsEq<ArgAbi<'a, Ty>>;
        let _: ::core::cmp::AssertParamIsEq<bool>;
        let _: ::core::cmp::AssertParamIsEq<u32>;
        let _: ::core::cmp::AssertParamIsEq<CanonAbi>;
    }
}Eq, #[automatically_derived]
impl<'a, Ty: ::core::hash::Hash> ::core::hash::Hash for FnAbi<'a, Ty> {
    #[inline]
    fn hash<__H: ::core::hash::Hasher>(&self, state: &mut __H) {
        ::core::hash::Hash::hash(&self.args, state);
        ::core::hash::Hash::hash(&self.ret, state);
        ::core::hash::Hash::hash(&self.c_variadic, state);
        ::core::hash::Hash::hash(&self.fixed_count, state);
        ::core::hash::Hash::hash(&self.conv, state);
        ::core::hash::Hash::hash(&self.can_unwind, state)
    }
}Hash, const _: () =
    {
        impl<'a, Ty> ::rustc_data_structures::stable_hash::StableHash for
            FnAbi<'a, Ty> where
            Ty: ::rustc_data_structures::stable_hash::StableHash {
            #[inline]
            fn stable_hash<__Hcx: ::rustc_data_structures::stable_hash::StableHashCtxt>(&self,
                __hcx: &mut __Hcx,
                __hasher:
                    &mut ::rustc_data_structures::stable_hash::StableHasher) {
                match *self {
                    FnAbi {
                        args: ref __binding_0,
                        ret: ref __binding_1,
                        c_variadic: ref __binding_2,
                        fixed_count: ref __binding_3,
                        conv: ref __binding_4,
                        can_unwind: ref __binding_5 } => {
                        { __binding_0.stable_hash(__hcx, __hasher); }
                        { __binding_1.stable_hash(__hcx, __hasher); }
                        { __binding_2.stable_hash(__hcx, __hasher); }
                        { __binding_3.stable_hash(__hcx, __hasher); }
                        { __binding_4.stable_hash(__hcx, __hasher); }
                        { __binding_5.stable_hash(__hcx, __hasher); }
                    }
                }
            }
        }
    };StableHash)]
604pub struct FnAbi<'a, Ty> {
605    /// The type, layout, and information about how each argument is passed.
606    pub args: Box<[ArgAbi<'a, Ty>]>,
607
608    /// The layout, type, and the way a value is returned from this function.
609    pub ret: ArgAbi<'a, Ty>,
610
611    /// Marks this function as variadic (accepting a variable number of arguments).
612    pub c_variadic: bool,
613
614    /// The count of non-variadic arguments.
615    ///
616    /// Should only be different from args.len() when c_variadic is true.
617    /// This can be used to know whether an argument is variadic or not.
618    pub fixed_count: u32,
619    /// The calling convention of this function.
620    pub conv: CanonAbi,
621    /// Indicates if an unwind may happen across a call to this function.
622    pub can_unwind: bool,
623}
624
625// Needs to be a custom impl because of the bounds on the `TyAndLayout` debug impl.
626impl<'a, Ty: fmt::Display> fmt::Debug for FnAbi<'a, Ty> {
627    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
628        let FnAbi { args, ret, c_variadic, fixed_count, conv, can_unwind } = self;
629        f.debug_struct("FnAbi")
630            .field("args", args)
631            .field("ret", ret)
632            .field("c_variadic", c_variadic)
633            .field("fixed_count", fixed_count)
634            .field("conv", conv)
635            .field("can_unwind", can_unwind)
636            .finish()
637    }
638}
639
640impl<'a, Ty> FnAbi<'a, Ty> {
641    pub fn adjust_for_foreign_abi<C>(&mut self, cx: &C, abi: ExternAbi)
642    where
643        Ty: TyAbiInterface<'a, C> + Copy,
644        C: HasDataLayout + HasTargetSpec + HasX86AbiOpt,
645    {
646        if abi == ExternAbi::X86Interrupt {
647            if let Some(arg) = self.args.first_mut() {
648                arg.pass_by_stack_offset(None);
649            }
650            return;
651        }
652
653        let spec = cx.target_spec();
654        match &spec.arch {
655            Arch::X86 => {
656                let (flavor, regparm) = match abi {
657                    ExternAbi::Fastcall { .. } | ExternAbi::Vectorcall { .. } => {
658                        (x86::Flavor::FastcallOrVectorcall, None)
659                    }
660                    ExternAbi::C { .. } | ExternAbi::Cdecl { .. } | ExternAbi::Stdcall { .. } => {
661                        (x86::Flavor::General, cx.x86_abi_opt().regparm)
662                    }
663                    _ => (x86::Flavor::General, None),
664                };
665                let reg_struct_return = cx.x86_abi_opt().reg_struct_return;
666                let opts = x86::X86Options { flavor, regparm, reg_struct_return };
667                if spec.is_like_msvc {
668                    x86_win32::compute_abi_info(cx, self, opts);
669                } else {
670                    x86::compute_abi_info(cx, self, opts);
671                }
672            }
673            Arch::X86_64 => match abi {
674                ExternAbi::SysV64 { .. } => x86_64::compute_abi_info(cx, self),
675                ExternAbi::Win64 { .. } | ExternAbi::Vectorcall { .. } => {
676                    x86_win64::compute_abi_info(cx, self)
677                }
678                _ => {
679                    if cx.target_spec().is_like_windows {
680                        x86_win64::compute_abi_info(cx, self)
681                    } else {
682                        x86_64::compute_abi_info(cx, self)
683                    }
684                }
685            },
686            Arch::AArch64 | Arch::Arm64EC => {
687                let kind = if cx.target_spec().is_like_darwin {
688                    aarch64::AbiKind::DarwinPCS
689                } else if cx.target_spec().is_like_windows {
690                    aarch64::AbiKind::Win64
691                } else {
692                    aarch64::AbiKind::AAPCS
693                };
694                aarch64::compute_abi_info(cx, self, kind)
695            }
696            Arch::AmdGpu => amdgpu::compute_abi_info(cx, self),
697            Arch::Arm => arm::compute_abi_info(cx, self),
698            Arch::Avr => avr::compute_abi_info(cx, self),
699            Arch::LoongArch32 | Arch::LoongArch64 => loongarch::compute_abi_info(cx, self),
700            Arch::M68k => m68k::compute_abi_info(cx, self),
701            Arch::CSky => csky::compute_abi_info(cx, self),
702            Arch::Mips | Arch::Mips32r6 => mips::compute_abi_info(cx, self),
703            Arch::Mips64 | Arch::Mips64r6 => mips64::compute_abi_info(cx, self),
704            Arch::PowerPC => powerpc::compute_abi_info(cx, self),
705            Arch::PowerPC64 => powerpc64::compute_abi_info(cx, self),
706            Arch::S390x => s390x::compute_abi_info(cx, self),
707            Arch::Msp430 => msp430::compute_abi_info(cx, self),
708            Arch::Sparc => sparc::compute_abi_info(cx, self),
709            Arch::Sparc64 => sparc64::compute_abi_info(cx, self),
710            Arch::Nvptx64 => {
711                if abi == ExternAbi::PtxKernel || abi == ExternAbi::GpuKernel {
712                    nvptx64::compute_ptx_kernel_abi_info(cx, self)
713                } else {
714                    nvptx64::compute_abi_info(cx, self)
715                }
716            }
717            Arch::Hexagon => hexagon::compute_abi_info(cx, self),
718            Arch::Xtensa => xtensa::compute_abi_info(cx, self),
719            Arch::RiscV32 | Arch::RiscV64 => riscv::compute_abi_info(cx, self),
720            Arch::Wasm32 | Arch::Wasm64 => wasm::compute_abi_info(cx, self),
721            Arch::Bpf => bpf::compute_abi_info(cx, self),
722            arch @ (Arch::SpirV | Arch::Other(_)) => {
723                {
    ::core::panicking::panic_fmt(format_args!("no lowering implemented for {0}",
            arch));
}panic!("no lowering implemented for {arch}")
724            }
725        }
726    }
727
728    pub fn adjust_for_rust_abi<C>(&mut self, cx: &C)
729    where
730        Ty: TyAbiInterface<'a, C> + Copy,
731        C: HasDataLayout + HasTargetSpec,
732    {
733        let spec = cx.target_spec();
734        match &spec.arch {
735            Arch::X86 => x86::compute_rust_abi_info(cx, self),
736            Arch::RiscV32 | Arch::RiscV64 => riscv::compute_rust_abi_info(cx, self),
737            Arch::LoongArch32 | Arch::LoongArch64 => loongarch::compute_rust_abi_info(cx, self),
738            Arch::AArch64 => aarch64::compute_rust_abi_info(cx, self),
739            Arch::Bpf => bpf::compute_rust_abi_info(self),
740            _ => {}
741        };
742
743        for (arg_idx, arg) in self
744            .args
745            .iter_mut()
746            .enumerate()
747            .map(|(idx, arg)| (Some(idx), arg))
748            .chain(iter::once((None, &mut self.ret)))
749        {
750            // If the logic above already picked a specific type to cast the argument to, leave that
751            // in place.
752            if #[allow(non_exhaustive_omitted_patterns)] match arg.mode {
    PassMode::Ignore | PassMode::Cast { .. } => true,
    _ => false,
}matches!(arg.mode, PassMode::Ignore | PassMode::Cast { .. }) {
753                continue;
754            }
755
756            if arg_idx.is_none()
757                && arg.layout.size > Primitive::Pointer(AddressSpace::ZERO).size(cx) * 2
758                && !#[allow(non_exhaustive_omitted_patterns)] match arg.layout.backend_repr {
    BackendRepr::SimdVector { .. } | BackendRepr::SimdScalableVector { .. } =>
        true,
    _ => false,
}matches!(
759                    arg.layout.backend_repr,
760                    BackendRepr::SimdVector { .. } | BackendRepr::SimdScalableVector { .. }
761                )
762            {
763                // Return values larger than 2 registers using a return area
764                // pointer. LLVM and Cranelift disagree about how to return
765                // values that don't fit in the registers designated for return
766                // values. LLVM will force the entire return value to be passed
767                // by return area pointer, while Cranelift will look at each IR level
768                // return value independently and decide to pass it in a
769                // register or not, which would result in the return value
770                // being passed partially in registers and partially through a
771                // return area pointer. For large IR-level values such as `i128`,
772                // cranelift will even split up the value into smaller chunks.
773                //
774                // While Cranelift may need to be fixed as the LLVM behavior is
775                // generally more correct with respect to the surface language,
776                // forcing this behavior in rustc itself makes it easier for
777                // other backends to conform to the Rust ABI and for the C ABI
778                // rustc already handles this behavior anyway.
779                //
780                // In addition LLVM's decision to pass the return value in
781                // registers or using a return area pointer depends on how
782                // exactly the return type is lowered to an LLVM IR type. For
783                // example `Option<u128>` can be lowered as `{ i128, i128 }`
784                // in which case the x86_64 backend would use a return area
785                // pointer, or it could be passed as `{ i32, i128 }` in which
786                // case the x86_64 backend would pass it in registers by taking
787                // advantage of an LLVM ABI extension that allows using 3
788                // registers for the x86_64 sysv call conv rather than the
789                // officially specified 2 registers.
790                //
791                // FIXME: Technically we should look at the amount of available
792                // return registers rather than guessing that there are 2
793                // registers for return values. In practice only a couple of
794                // architectures have less than 2 return registers. None of
795                // which supported by Cranelift.
796                //
797                // NOTE: This adjustment is only necessary for the Rust ABI as
798                // for other ABI's the calling convention implementations in
799                // rustc_target already ensure any return value which doesn't
800                // fit in the available amount of return registers is passed in
801                // the right way for the current target.
802                //
803                // The adjustment is not necessary nor desired for types with a vector
804                // representation; those are handled below.
805                arg.make_indirect();
806                continue;
807            }
808
809            match arg.layout.backend_repr {
810                BackendRepr::Memory { .. } => {
811                    // Compute `Aggregate` ABI.
812
813                    let is_indirect_not_on_stack =
814                        #[allow(non_exhaustive_omitted_patterns)] match arg.mode {
    PassMode::Indirect { on_stack: false, .. } => true,
    _ => false,
}matches!(arg.mode, PassMode::Indirect { on_stack: false, .. });
815                    if !is_indirect_not_on_stack {
    ::core::panicking::panic("assertion failed: is_indirect_not_on_stack")
};assert!(is_indirect_not_on_stack);
816
817                    let size = arg.layout.size;
818                    if arg.layout.is_sized()
819                        && size <= Primitive::Pointer(AddressSpace::ZERO).size(cx)
820                    {
821                        // We want to pass small aggregates as immediates, but using
822                        // an LLVM aggregate type for this leads to bad optimizations,
823                        // so we pick an appropriately sized integer type instead.
824                        let attr = if layout_is_noundef(arg.layout, cx) {
825                            ArgAttribute::NoUndef
826                        } else {
827                            ArgAttribute::default()
828                        };
829                        arg.cast_to_with_attrs(Reg { kind: RegKind::Integer, size }, attr.into());
830                    }
831                }
832
833                BackendRepr::SimdVector { .. } => {
834                    // This is a fun case! The gist of what this is doing is
835                    // that we want callers and callees to always agree on the
836                    // ABI of how they pass SIMD arguments. If we were to *not*
837                    // make these arguments indirect then they'd be immediates
838                    // in LLVM, which means that they'd used whatever the
839                    // appropriate ABI is for the callee and the caller. That
840                    // means, for example, if the caller doesn't have AVX
841                    // enabled but the callee does, then passing an AVX argument
842                    // across this boundary would cause corrupt data to show up.
843                    //
844                    // This problem is fixed by unconditionally passing SIMD
845                    // arguments through memory between callers and callees
846                    // which should get them all to agree on ABI regardless of
847                    // target feature sets. Some more information about this
848                    // issue can be found in #44367.
849                    //
850                    // We *could* do better in some cases, e.g. on x86_64 targets where SSE2 is
851                    // required. However, it turns out that that makes LLVM worse at optimizing this
852                    // code, so we pass things indirectly even there. See #139029 for more on that.
853                    if spec.simd_types_indirect {
854                        arg.make_indirect();
855                    }
856                }
857
858                _ => {}
859            }
860        }
861    }
862}
863
864/// Determines whether `layout` contains no uninit bytes (no padding, no unions),
865/// using only the computed layout.
866///
867/// Conservative: returns `false` for anything it cannot prove fully initialized,
868/// including multi-variant enums and SIMD vectors.
869// FIXME: extend to multi-variant enums (per-variant padding analysis needed).
870fn layout_is_noundef<'a, Ty, C>(layout: TyAndLayout<'a, Ty>, cx: &C) -> bool
871where
872    Ty: TyAbiInterface<'a, C> + Copy,
873    C: HasDataLayout,
874{
875    match layout.backend_repr {
876        BackendRepr::Scalar(scalar) => !scalar.is_uninit_valid(),
877        BackendRepr::ScalarPair(s1, s2) => {
878            !s1.is_uninit_valid()
879                && !s2.is_uninit_valid()
880                // Ensure there is no padding.
881                && s1.size(cx) + s2.size(cx) == layout.size
882        }
883        BackendRepr::Memory { .. } => match layout.fields {
884            FieldsShape::Primitive | FieldsShape::Union(_) => false,
885            // Array elements are at stride offsets with no inter-element gaps.
886            FieldsShape::Array { stride: _, count } => {
887                count == 0 || layout_is_noundef(layout.field(cx, 0), cx)
888            }
889            FieldsShape::Arbitrary { .. } => {
890                // With `Variants::Multiple`, `layout.fields` only covers shared
891                // bytes (niche/discriminant); per-variant data is absent, so
892                // full coverage cannot be proven.
893                #[allow(non_exhaustive_omitted_patterns)] match layout.variants {
    Variants::Single { .. } => true,
    _ => false,
}matches!(layout.variants, Variants::Single { .. }) && fields_are_noundef(layout, cx)
894            }
895        },
896        BackendRepr::SimdVector { .. } | BackendRepr::SimdScalableVector { .. } => false,
897    }
898}
899
900/// Returns `true` if the fields of `layout` contiguously cover bytes `0..layout.size`
901/// with no padding gaps and each field is recursively `layout_is_noundef`.
902fn fields_are_noundef<'a, Ty, C>(layout: TyAndLayout<'a, Ty>, cx: &C) -> bool
903where
904    Ty: TyAbiInterface<'a, C> + Copy,
905    C: HasDataLayout,
906{
907    let mut cursor = Size::ZERO;
908    for i in layout.fields.index_by_increasing_offset() {
909        let field = layout.field(cx, i);
910        if field.size == Size::ZERO {
911            continue;
912        }
913        if layout.fields.offset(i) != cursor {
914            return false;
915        }
916        if !layout_is_noundef(field, cx) {
917            return false;
918        }
919        cursor += field.size;
920    }
921    cursor == layout.size
922}
923
924// Some types are used a lot. Make sure they don't unintentionally get bigger.
925#[cfg(target_pointer_width = "64")]
926mod size_asserts {
927    use rustc_data_structures::static_assert_size;
928
929    use super::*;
930    // tidy-alphabetical-start
931    const _: [(); 56] = [(); ::std::mem::size_of::<ArgAbi<'_, usize>>()];static_assert_size!(ArgAbi<'_, usize>, 56);
932    const _: [(); 80] = [(); ::std::mem::size_of::<FnAbi<'_, usize>>()];static_assert_size!(FnAbi<'_, usize>, 80);
933    // tidy-alphabetical-end
934}