1#[cfg(feature = "nightly")]
2use crate::{BackendRepr, FieldsShape, Primitive, Size, TyAbiInterface, TyAndLayout, Variants};
3
4mod reg;
5
6pub use reg::{Reg, RegKind};
7
8#[derive(#[automatically_derived]
impl ::core::marker::Copy for HomogeneousAggregate { }Copy, #[automatically_derived]
impl ::core::clone::Clone for HomogeneousAggregate {
#[inline]
fn clone(&self) -> HomogeneousAggregate {
let _: ::core::clone::AssertParamIsClone<Reg>;
*self
}
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for HomogeneousAggregate {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
HomogeneousAggregate::Homogeneous(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"Homogeneous", &__self_0),
HomogeneousAggregate::NoData =>
::core::fmt::Formatter::write_str(f, "NoData"),
}
}
}Debug)]
10pub enum HomogeneousAggregate {
11 Homogeneous(Reg),
14
15 NoData,
17}
18
19#[derive(#[automatically_derived]
impl ::core::marker::Copy for Heterogeneous { }Copy, #[automatically_derived]
impl ::core::clone::Clone for Heterogeneous {
#[inline]
fn clone(&self) -> Heterogeneous { *self }
}Clone, #[automatically_derived]
impl ::core::fmt::Debug for Heterogeneous {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f, "Heterogeneous")
}
}Debug)]
23pub struct Heterogeneous;
24
25impl HomogeneousAggregate {
26 pub fn unit(self) -> Option<Reg> {
29 match self {
30 HomogeneousAggregate::Homogeneous(reg) => Some(reg),
31 HomogeneousAggregate::NoData => None,
32 }
33 }
34
35 #[cfg(feature = "nightly")]
39 fn merge(self, other: HomogeneousAggregate) -> Result<HomogeneousAggregate, Heterogeneous> {
40 match (self, other) {
41 (x, HomogeneousAggregate::NoData) | (HomogeneousAggregate::NoData, x) => Ok(x),
42
43 (HomogeneousAggregate::Homogeneous(a), HomogeneousAggregate::Homogeneous(b)) => {
44 if a != b {
45 return Err(Heterogeneous);
46 }
47 Ok(self)
48 }
49 }
50 }
51}
52
53#[cfg(feature = "nightly")]
54impl<'a, Ty> TyAndLayout<'a, Ty> {
55 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("homogeneous_aggregate",
"rustc_abi::callconv", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_abi/src/callconv.rs"),
::tracing_core::__macro_support::Option::Some(64u32),
::tracing_core::__macro_support::Option::Some("rustc_abi::callconv"),
::tracing_core::field::FieldSet::new(&["self"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&self)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return:
Result<HomogeneousAggregate, Heterogeneous> = loop {};
return __tracing_attr_fake_return;
}
{
match self.backend_repr {
BackendRepr::Scalar(scalar) => {
let kind =
match scalar.primitive() {
Primitive::Int(..) | Primitive::Pointer(_) =>
RegKind::Integer,
Primitive::Float(_) => RegKind::Float,
};
Ok(HomogeneousAggregate::Homogeneous(Reg {
kind,
size: self.size,
}))
}
BackendRepr::SimdVector { element, count: _ } => {
if !!self.is_zst() {
::core::panicking::panic("assertion failed: !self.is_zst()")
};
Ok(HomogeneousAggregate::Homogeneous(Reg {
kind: RegKind::Vector {
hint_vector_elem: element.primitive(),
},
size: self.size,
}))
}
BackendRepr::SimdScalableVector { .. } => {
{
::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
format_args!("`homogeneous_aggregate` should not be called for scalable vectors")));
}
}
BackendRepr::ScalarPair(..) | BackendRepr::Memory {
sized: true } => {
let from_fields_at =
|layout: Self, start: Size|
-> Result<(HomogeneousAggregate, Size), Heterogeneous>
{
let is_union =
match layout.fields {
FieldsShape::Primitive => {
{
::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
format_args!("aggregates can\'t have `FieldsShape::Primitive`")));
}
}
FieldsShape::Array { count, .. } => {
match (&start, &Size::ZERO) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
let result =
if count > 0 {
layout.field(cx, 0).homogeneous_aggregate(cx)?
} else { HomogeneousAggregate::NoData };
return Ok((result, layout.size));
}
FieldsShape::Union(_) => true,
FieldsShape::Arbitrary { .. } => false,
};
let mut result = HomogeneousAggregate::NoData;
let mut total = start;
for i in 0..layout.fields.count() {
let field = layout.field(cx, i);
if field.is_1zst() { continue; }
if !is_union && total != layout.fields.offset(i) {
return Err(Heterogeneous);
}
result = result.merge(field.homogeneous_aggregate(cx)?)?;
let size = field.size;
if is_union {
total = total.max(size);
} else { total += size; }
}
Ok((result, total))
};
let (mut result, mut total) =
from_fields_at(*self, Size::ZERO)?;
match &self.variants {
Variants::Single { .. } | Variants::Empty => {}
Variants::Multiple { variants, .. } => {
let variant_start = total;
for variant_idx in variants.indices() {
let (variant_result, variant_total) =
from_fields_at(self.for_variant(cx, variant_idx),
variant_start)?;
result = result.merge(variant_result)?;
total = total.max(variant_total);
}
}
}
if total != self.size {
Err(Heterogeneous)
} else {
match result {
HomogeneousAggregate::Homogeneous(_) => {
match (&total, &Size::ZERO) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
}
HomogeneousAggregate::NoData => {
match (&total, &Size::ZERO) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
}
}
Ok(result)
}
}
BackendRepr::Memory { sized: false } => Err(Heterogeneous),
}
}
}
}#[tracing::instrument(skip(cx), level = "debug")]
65 pub fn homogeneous_aggregate<C>(&self, cx: &C) -> Result<HomogeneousAggregate, Heterogeneous>
66 where
67 Ty: TyAbiInterface<'a, C> + Copy,
68 {
69 match self.backend_repr {
70 BackendRepr::Scalar(scalar) => {
72 let kind = match scalar.primitive() {
73 Primitive::Int(..) | Primitive::Pointer(_) => RegKind::Integer,
74 Primitive::Float(_) => RegKind::Float,
75 };
76 Ok(HomogeneousAggregate::Homogeneous(Reg { kind, size: self.size }))
77 }
78
79 BackendRepr::SimdVector { element, count: _ } => {
80 assert!(!self.is_zst());
81
82 Ok(HomogeneousAggregate::Homogeneous(Reg {
83 kind: RegKind::Vector { hint_vector_elem: element.primitive() },
84 size: self.size,
85 }))
86 }
87
88 BackendRepr::SimdScalableVector { .. } => {
89 unreachable!("`homogeneous_aggregate` should not be called for scalable vectors")
90 }
91
92 BackendRepr::ScalarPair(..) | BackendRepr::Memory { sized: true } => {
93 let from_fields_at =
96 |layout: Self,
97 start: Size|
98 -> Result<(HomogeneousAggregate, Size), Heterogeneous> {
99 let is_union = match layout.fields {
100 FieldsShape::Primitive => {
101 unreachable!("aggregates can't have `FieldsShape::Primitive`")
102 }
103 FieldsShape::Array { count, .. } => {
104 assert_eq!(start, Size::ZERO);
105
106 let result = if count > 0 {
107 layout.field(cx, 0).homogeneous_aggregate(cx)?
108 } else {
109 HomogeneousAggregate::NoData
110 };
111 return Ok((result, layout.size));
112 }
113 FieldsShape::Union(_) => true,
114 FieldsShape::Arbitrary { .. } => false,
115 };
116
117 let mut result = HomogeneousAggregate::NoData;
118 let mut total = start;
119
120 for i in 0..layout.fields.count() {
121 let field = layout.field(cx, i);
122 if field.is_1zst() {
123 continue;
126 }
127
128 if !is_union && total != layout.fields.offset(i) {
129 return Err(Heterogeneous);
131 }
132
133 result = result.merge(field.homogeneous_aggregate(cx)?)?;
134
135 let size = field.size;
137 if is_union {
138 total = total.max(size);
139 } else {
140 total += size;
141 }
142 }
143
144 Ok((result, total))
145 };
146
147 let (mut result, mut total) = from_fields_at(*self, Size::ZERO)?;
148
149 match &self.variants {
150 Variants::Single { .. } | Variants::Empty => {}
151 Variants::Multiple { variants, .. } => {
152 let variant_start = total;
165 for variant_idx in variants.indices() {
166 let (variant_result, variant_total) =
167 from_fields_at(self.for_variant(cx, variant_idx), variant_start)?;
168
169 result = result.merge(variant_result)?;
170 total = total.max(variant_total);
171 }
172 }
173 }
174
175 if total != self.size {
177 Err(Heterogeneous)
178 } else {
179 match result {
180 HomogeneousAggregate::Homogeneous(_) => {
181 assert_ne!(total, Size::ZERO);
182 }
183 HomogeneousAggregate::NoData => {
184 assert_eq!(total, Size::ZERO);
185 }
186 }
187 Ok(result)
188 }
189 }
190 BackendRepr::Memory { sized: false } => Err(Heterogeneous),
191 }
192 }
193}