1use rustc_abi::Align;
2use rustc_middle::mir::*;
3use rustc_middle::ty::{self, AdtDef, TyCtxt};
4use tracing::debug;
5
6pub fn place_unalignment<'tcx, L>(
10 tcx: TyCtxt<'tcx>,
11 local_decls: &L,
12 typing_env: ty::TypingEnv<'tcx>,
13 place: Place<'tcx>,
14) -> Option<(AdtDef<'tcx>, Align)>
15where
16 L: HasLocalDecls<'tcx>,
17{
18 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/util/alignment.rs:18",
"rustc_const_eval::util::alignment",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/util/alignment.rs"),
::tracing_core::__macro_support::Option::Some(18u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::util::alignment"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("unalignment({0:?})",
place) as &dyn Value))])
});
} else { ; }
};debug!("unalignment({:?})", place);
19 let Some((descr, pack)) = most_packed_projection(tcx, local_decls, place) else {
20 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/util/alignment.rs:20",
"rustc_const_eval::util::alignment",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/util/alignment.rs"),
::tracing_core::__macro_support::Option::Some(20u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::util::alignment"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("unalignment({0:?}) - not within packed",
place) as &dyn Value))])
});
} else { ; }
};debug!("unalignment({:?}) - not within packed", place);
21 return None;
22 };
23
24 let ty = place.ty(local_decls, tcx).ty;
25 let unsized_tail = || tcx.struct_tail_for_codegen(ty, typing_env);
26 match tcx.layout_of(typing_env.as_query_input(ty)) {
27 Ok(layout)
28 if layout.align.abi <= pack
29 && (layout.is_sized()
30 || #[allow(non_exhaustive_omitted_patterns)] match unsized_tail().kind() {
ty::Slice(..) | ty::Str => true,
_ => false,
}matches!(unsized_tail().kind(), ty::Slice(..) | ty::Str)) =>
31 {
32 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/util/alignment.rs:37",
"rustc_const_eval::util::alignment",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/util/alignment.rs"),
::tracing_core::__macro_support::Option::Some(37u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::util::alignment"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("unalignment({0:?}) - align = {1}, packed = {2}; not unaligned",
place, layout.align.bytes(), pack.bytes()) as &dyn Value))])
});
} else { ; }
};debug!(
38 "unalignment({:?}) - align = {}, packed = {}; not unaligned",
39 place,
40 layout.align.bytes(),
41 pack.bytes()
42 );
43 None
44 }
45 _ => {
46 {
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_const_eval/src/util/alignment.rs:47",
"rustc_const_eval::util::alignment",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_const_eval/src/util/alignment.rs"),
::tracing_core::__macro_support::Option::Some(47u32),
::tracing_core::__macro_support::Option::Some("rustc_const_eval::util::alignment"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("unalignment({0:?}) - unaligned",
place) as &dyn Value))])
});
} else { ; }
};debug!("unalignment({:?}) - unaligned", place);
48 Some((descr, pack))
49 }
50 }
51}
52
53pub fn most_packed_projection<'tcx, L>(
57 tcx: TyCtxt<'tcx>,
58 local_decls: &L,
59 place: Place<'tcx>,
60) -> Option<(AdtDef<'tcx>, Align)>
61where
62 L: HasLocalDecls<'tcx>,
63{
64 place
65 .iter_projections()
66 .rev()
67 .take_while(|(_base, elem)| !#[allow(non_exhaustive_omitted_patterns)] match elem {
ProjectionElem::Deref => true,
_ => false,
}matches!(elem, ProjectionElem::Deref))
69 .filter_map(|(base, _elem)| {
71 let adt = base.ty(local_decls, tcx).ty.ty_adt_def()?;
72 let pack = adt.repr().pack?;
73 Some((adt, pack))
74 })
75 .min_by_key(|(_, align)| *align)
78}