2022-10-23 08:12:05 +00:00
|
|
|
//! Compute the binary representation of structs, unions and enums
|
|
|
|
|
2023-04-16 10:21:12 +00:00
|
|
|
use std::{cmp, ops::Bound};
|
2022-10-23 08:12:05 +00:00
|
|
|
|
|
|
|
use hir_def::{
|
2023-04-06 17:23:29 +00:00
|
|
|
data::adt::VariantData,
|
2023-04-16 10:21:12 +00:00
|
|
|
layout::{Integer, LayoutCalculator, ReprOptions, TargetDataLayout},
|
2023-05-18 07:59:03 +00:00
|
|
|
AdtId, EnumVariantId, LocalEnumVariantId, VariantId,
|
2022-10-23 08:12:05 +00:00
|
|
|
};
|
2022-12-06 22:29:38 +00:00
|
|
|
use la_arena::RawIdx;
|
2022-12-09 09:09:55 +00:00
|
|
|
use smallvec::SmallVec;
|
2023-05-25 21:15:37 +00:00
|
|
|
use triomphe::Arc;
|
2022-10-23 08:12:05 +00:00
|
|
|
|
2023-04-16 10:21:12 +00:00
|
|
|
use crate::{
|
|
|
|
db::HirDatabase,
|
|
|
|
lang_items::is_unsafe_cell,
|
|
|
|
layout::{field_ty, Layout, LayoutError, RustcEnumVariantIdx},
|
2023-07-20 09:38:38 +00:00
|
|
|
Substitution, TraitEnvironment,
|
2023-04-16 10:21:12 +00:00
|
|
|
};
|
2022-10-27 19:58:34 +00:00
|
|
|
|
2023-05-25 21:15:37 +00:00
|
|
|
use super::LayoutCx;
|
2022-10-23 08:12:05 +00:00
|
|
|
|
2022-12-06 22:29:38 +00:00
|
|
|
pub(crate) fn struct_variant_idx() -> RustcEnumVariantIdx {
|
|
|
|
RustcEnumVariantIdx(LocalEnumVariantId::from_raw(RawIdx::from(0)))
|
|
|
|
}
|
2022-10-23 08:12:05 +00:00
|
|
|
|
|
|
|
pub fn layout_of_adt_query(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: AdtId,
|
|
|
|
subst: Substitution,
|
2023-07-20 09:38:38 +00:00
|
|
|
trait_env: Arc<TraitEnvironment>,
|
2023-05-25 21:15:37 +00:00
|
|
|
) -> Result<Arc<Layout>, LayoutError> {
|
2023-07-20 09:38:38 +00:00
|
|
|
let krate = trait_env.krate;
|
2023-07-03 18:34:09 +00:00
|
|
|
let Some(target) = db.target_data_layout(krate) else {
|
|
|
|
return Err(LayoutError::TargetLayoutNotAvailable);
|
|
|
|
};
|
2023-07-20 09:38:38 +00:00
|
|
|
let cx = LayoutCx { target: &target };
|
2022-12-21 14:11:24 +00:00
|
|
|
let dl = cx.current_data_layout();
|
2022-10-23 08:12:05 +00:00
|
|
|
let handle_variant = |def: VariantId, var: &VariantData| {
|
|
|
|
var.fields()
|
|
|
|
.iter()
|
2023-07-20 09:38:38 +00:00
|
|
|
.map(|(fd, _)| db.layout_of_ty(field_ty(db, def, fd, &subst), trait_env.clone()))
|
2022-10-23 08:12:05 +00:00
|
|
|
.collect::<Result<Vec<_>, _>>()
|
|
|
|
};
|
2023-05-15 09:59:03 +00:00
|
|
|
let (variants, repr) = match def {
|
2022-10-23 08:12:05 +00:00
|
|
|
AdtId::StructId(s) => {
|
|
|
|
let data = db.struct_data(s);
|
2022-12-09 09:09:55 +00:00
|
|
|
let mut r = SmallVec::<[_; 1]>::new();
|
2022-12-06 22:29:38 +00:00
|
|
|
r.push(handle_variant(s.into(), &data.variant_data)?);
|
2023-05-15 09:59:03 +00:00
|
|
|
(r, data.repr.unwrap_or_default())
|
2022-12-06 22:29:38 +00:00
|
|
|
}
|
|
|
|
AdtId::UnionId(id) => {
|
|
|
|
let data = db.union_data(id);
|
2022-12-09 09:09:55 +00:00
|
|
|
let mut r = SmallVec::new();
|
2022-12-06 22:29:38 +00:00
|
|
|
r.push(handle_variant(id.into(), &data.variant_data)?);
|
2023-05-15 09:59:03 +00:00
|
|
|
(r, data.repr.unwrap_or_default())
|
2022-10-23 08:12:05 +00:00
|
|
|
}
|
|
|
|
AdtId::EnumId(e) => {
|
|
|
|
let data = db.enum_data(e);
|
|
|
|
let r = data
|
|
|
|
.variants
|
|
|
|
.iter()
|
|
|
|
.map(|(idx, v)| {
|
2022-12-06 22:29:38 +00:00
|
|
|
handle_variant(
|
|
|
|
EnumVariantId { parent: e, local_id: idx }.into(),
|
|
|
|
&v.variant_data,
|
|
|
|
)
|
2022-10-23 08:12:05 +00:00
|
|
|
})
|
2022-12-09 09:09:55 +00:00
|
|
|
.collect::<Result<SmallVec<_>, _>>()?;
|
2023-05-15 09:59:03 +00:00
|
|
|
(r, data.repr.unwrap_or_default())
|
2022-10-23 08:12:05 +00:00
|
|
|
}
|
|
|
|
};
|
2023-05-25 21:15:37 +00:00
|
|
|
let variants = variants
|
|
|
|
.iter()
|
2023-07-06 14:03:17 +00:00
|
|
|
.map(|it| it.iter().map(|it| &**it).collect::<Vec<_>>())
|
2023-05-25 21:15:37 +00:00
|
|
|
.collect::<SmallVec<[_; 1]>>();
|
2023-07-06 14:03:17 +00:00
|
|
|
let variants = variants.iter().map(|it| it.iter().collect()).collect();
|
2023-05-25 21:15:37 +00:00
|
|
|
let result = if matches!(def, AdtId::UnionId(..)) {
|
|
|
|
cx.layout_of_union(&repr, &variants).ok_or(LayoutError::Unknown)?
|
2022-10-23 08:12:05 +00:00
|
|
|
} else {
|
2022-12-06 22:29:38 +00:00
|
|
|
cx.layout_of_struct_or_enum(
|
|
|
|
&repr,
|
|
|
|
&variants,
|
2023-05-15 09:59:03 +00:00
|
|
|
matches!(def, AdtId::EnumId(..)),
|
2023-03-30 07:37:52 +00:00
|
|
|
is_unsafe_cell(db, def),
|
2022-12-06 22:29:38 +00:00
|
|
|
layout_scalar_valid_range(db, def),
|
2023-04-16 10:21:12 +00:00
|
|
|
|min, max| repr_discr(&dl, &repr, min, max).unwrap_or((Integer::I8, false)),
|
2022-12-06 22:29:38 +00:00
|
|
|
variants.iter_enumerated().filter_map(|(id, _)| {
|
|
|
|
let AdtId::EnumId(e) = def else { return None };
|
2023-02-03 11:16:25 +00:00
|
|
|
let d =
|
|
|
|
db.const_eval_discriminant(EnumVariantId { parent: e, local_id: id.0 }).ok()?;
|
2022-12-06 22:29:38 +00:00
|
|
|
Some((id, d))
|
|
|
|
}),
|
|
|
|
// FIXME: The current code for niche-filling relies on variant indices
|
|
|
|
// instead of actual discriminants, so enums with
|
|
|
|
// explicit discriminants (RFC #2363) would misbehave and we should disable
|
|
|
|
// niche optimization for them.
|
|
|
|
// The code that do it in rustc:
|
|
|
|
// repr.inhibit_enum_layout_opt() || def
|
|
|
|
// .variants()
|
|
|
|
// .iter_enumerated()
|
|
|
|
// .any(|(i, v)| v.discr != ty::VariantDiscr::Relative(i.as_u32()))
|
|
|
|
repr.inhibit_enum_layout_opt(),
|
2023-05-15 09:59:03 +00:00
|
|
|
!matches!(def, AdtId::EnumId(..))
|
2022-12-06 22:29:38 +00:00
|
|
|
&& variants
|
|
|
|
.iter()
|
|
|
|
.next()
|
2023-07-06 14:03:17 +00:00
|
|
|
.and_then(|it| it.last().map(|it| !it.is_unsized()))
|
2022-12-06 22:29:38 +00:00
|
|
|
.unwrap_or(true),
|
|
|
|
)
|
2023-05-25 21:15:37 +00:00
|
|
|
.ok_or(LayoutError::SizeOverflow)?
|
|
|
|
};
|
|
|
|
Ok(Arc::new(result))
|
2022-10-23 08:12:05 +00:00
|
|
|
}
|
|
|
|
|
2022-10-27 19:58:34 +00:00
|
|
|
fn layout_scalar_valid_range(db: &dyn HirDatabase, def: AdtId) -> (Bound<u128>, Bound<u128>) {
|
|
|
|
let attrs = db.attrs(def.into());
|
|
|
|
let get = |name| {
|
|
|
|
let attr = attrs.by_key(name).tt_values();
|
|
|
|
for tree in attr {
|
2023-07-06 14:03:17 +00:00
|
|
|
if let Some(it) = tree.token_trees.first() {
|
2023-09-29 13:30:47 +00:00
|
|
|
let text = it.to_string().replace('_', "");
|
|
|
|
let base = match text.as_bytes() {
|
|
|
|
[b'0', b'x', ..] => 16,
|
|
|
|
[b'0', b'o', ..] => 8,
|
|
|
|
[b'0', b'b', ..] => 2,
|
|
|
|
_ => 10,
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Ok(it) = u128::from_str_radix(&text, base) {
|
2023-07-06 14:03:17 +00:00
|
|
|
return Bound::Included(it);
|
2022-10-27 19:58:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Bound::Unbounded
|
|
|
|
};
|
|
|
|
(get("rustc_layout_scalar_valid_range_start"), get("rustc_layout_scalar_valid_range_end"))
|
|
|
|
}
|
|
|
|
|
2022-10-23 08:12:05 +00:00
|
|
|
pub fn layout_of_adt_recover(
|
|
|
|
_: &dyn HirDatabase,
|
|
|
|
_: &[String],
|
|
|
|
_: &AdtId,
|
|
|
|
_: &Substitution,
|
2023-07-20 09:38:38 +00:00
|
|
|
_: &Arc<TraitEnvironment>,
|
2023-05-25 21:15:37 +00:00
|
|
|
) -> Result<Arc<Layout>, LayoutError> {
|
2022-10-23 08:12:05 +00:00
|
|
|
user_error!("infinite sized recursive type");
|
|
|
|
}
|
2023-04-16 10:21:12 +00:00
|
|
|
|
|
|
|
/// Finds the appropriate Integer type and signedness for the given
|
|
|
|
/// signed discriminant range and `#[repr]` attribute.
|
|
|
|
/// N.B.: `u128` values above `i128::MAX` will be treated as signed, but
|
|
|
|
/// that shouldn't affect anything, other than maybe debuginfo.
|
|
|
|
fn repr_discr(
|
|
|
|
dl: &TargetDataLayout,
|
|
|
|
repr: &ReprOptions,
|
|
|
|
min: i128,
|
|
|
|
max: i128,
|
|
|
|
) -> Result<(Integer, bool), LayoutError> {
|
|
|
|
// Theoretically, negative values could be larger in unsigned representation
|
|
|
|
// than the unsigned representation of the signed minimum. However, if there
|
|
|
|
// are any negative values, the only valid unsigned representation is u128
|
|
|
|
// which can fit all i128 values, so the result remains unaffected.
|
|
|
|
let unsigned_fit = Integer::fit_unsigned(cmp::max(min as u128, max as u128));
|
|
|
|
let signed_fit = cmp::max(Integer::fit_signed(min), Integer::fit_signed(max));
|
|
|
|
|
|
|
|
if let Some(ity) = repr.int {
|
|
|
|
let discr = Integer::from_attr(dl, ity);
|
|
|
|
let fit = if ity.is_signed() { signed_fit } else { unsigned_fit };
|
|
|
|
if discr < fit {
|
|
|
|
return Err(LayoutError::UserError(
|
|
|
|
"Integer::repr_discr: `#[repr]` hint too small for \
|
|
|
|
discriminant range of enum "
|
2023-09-09 20:45:01 +00:00
|
|
|
.into(),
|
2023-04-16 10:21:12 +00:00
|
|
|
));
|
|
|
|
}
|
|
|
|
return Ok((discr, ity.is_signed()));
|
|
|
|
}
|
|
|
|
|
|
|
|
let at_least = if repr.c() {
|
|
|
|
// This is usually I32, however it can be different on some platforms,
|
|
|
|
// notably hexagon and arm-none/thumb-none
|
|
|
|
dl.c_enum_min_size
|
|
|
|
} else {
|
|
|
|
// repr(Rust) enums try to be as small as possible
|
|
|
|
Integer::I8
|
|
|
|
};
|
|
|
|
|
|
|
|
// If there are no negative values, we can use the unsigned fit.
|
|
|
|
Ok(if min >= 0 {
|
|
|
|
(cmp::max(unsigned_fit, at_least), false)
|
|
|
|
} else {
|
|
|
|
(cmp::max(signed_fit, at_least), true)
|
|
|
|
})
|
|
|
|
}
|