1use rustc_middle::mir::{self, NonDivergingIntrinsic, StmtDebugInfo};
2use rustc_middle::{bug, span_bug, ty};
3use tracing::instrument;
4
5use super::{FunctionCx, LocalRef};
6use crate::mir::retag;
7use crate::traits::*;
8
9impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
10 #[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("codegen_statement",
"rustc_codegen_ssa::mir::statement",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_codegen_ssa/src/mir/statement.rs"),
::tracing_core::__macro_support::Option::Some(10u32),
::tracing_core::__macro_support::Option::Some("rustc_codegen_ssa::mir::statement"),
::tracing_core::field::FieldSet::new(&["statement"],
::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(&statement)
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: () = loop {};
return __tracing_attr_fake_return;
}
{
self.codegen_stmt_debuginfos(bx, &statement.debuginfos);
self.set_debug_loc(bx, statement.source_info);
match statement.kind {
mir::StatementKind::Assign((ref place, ref rvalue)) => {
let needs_retag =
bx.tcx().sess.opts.unstable_opts.codegen_emit_retag.is_some()
&& retag::rvalue_needs_retag(rvalue);
if let Some(index) = place.as_local() {
match self.locals[index] {
LocalRef::Place(cg_dest) => {
self.codegen_rvalue(bx, cg_dest, rvalue);
if needs_retag {
self.codegen_retag_place(bx, cg_dest, false);
}
}
LocalRef::UnsizedPlace(cg_indirect_dest) => {
let ty = cg_indirect_dest.layout.ty;
::rustc_middle::util::bug::span_bug_fmt(statement.source_info.span,
format_args!("cannot reallocate from `UnsizedPlace({0})` into `{1:?}`; dynamic alloca is not supported",
ty, rvalue));
}
LocalRef::PendingOperand => {
let mut operand = self.codegen_rvalue_operand(bx, rvalue);
if needs_retag {
operand = self.codegen_retag_operand(bx, operand, false);
}
self.overwrite_local(index, LocalRef::Operand(operand));
self.debug_introduce_local(bx, index);
}
LocalRef::Operand(op) => {
if !op.layout.is_zst() {
::rustc_middle::util::bug::span_bug_fmt(statement.source_info.span,
format_args!("operand {0:?} already assigned", rvalue));
}
let operand = self.codegen_rvalue_operand(bx, rvalue);
if needs_retag {
self.codegen_retag_operand(bx, operand, false);
}
}
}
} else {
let cg_dest = self.codegen_place(bx, place.as_ref());
self.codegen_rvalue(bx, cg_dest, rvalue);
if needs_retag {
self.codegen_retag_place(bx, cg_dest, false);
}
}
}
mir::StatementKind::SetDiscriminant { ref place, variant_index
} => {
self.codegen_place(bx,
(**place).as_ref()).codegen_set_discr(bx, variant_index);
}
mir::StatementKind::StorageLive(local) => {
if let LocalRef::Place(cg_place) = self.locals[local] {
cg_place.storage_live(bx);
} else if let LocalRef::UnsizedPlace(cg_indirect_place) =
self.locals[local] {
cg_indirect_place.storage_live(bx);
}
}
mir::StatementKind::StorageDead(local) => {
if let LocalRef::Place(cg_place) = self.locals[local] {
cg_place.storage_dead(bx);
} else if let LocalRef::UnsizedPlace(cg_indirect_place) =
self.locals[local] {
cg_indirect_place.storage_dead(bx);
}
}
mir::StatementKind::Coverage(ref kind) => {
self.codegen_coverage(bx, kind,
statement.source_info.scope);
}
mir::StatementKind::Intrinsic(NonDivergingIntrinsic::Assume(ref op))
=> {
let op_val = self.codegen_operand(bx, op);
bx.assume(op_val.immediate());
}
mir::StatementKind::Intrinsic(NonDivergingIntrinsic::CopyNonOverlapping(mir::CopyNonOverlapping {
ref count, ref src, ref dst })) => {
let dst_val = self.codegen_operand(bx, dst);
let src_val = self.codegen_operand(bx, src);
let count = self.codegen_operand(bx, count).immediate();
let &ty::RawPtr(pointee, _) =
dst_val.layout.ty.kind() else {
::rustc_middle::util::bug::bug_fmt(format_args!("expected pointer"))
};
let pointee_layout =
bx.tcx().layout_of(bx.typing_env().as_query_input(pointee)).expect("expected pointee to have a layout");
let elem_size = pointee_layout.layout.size().bytes();
let bytes =
bx.unchecked_sumul(count, bx.const_usize(elem_size));
let align = pointee_layout.layout.align.abi;
let dst = dst_val.immediate();
let src = src_val.immediate();
bx.memcpy(dst, align, src, align, bytes,
crate::MemFlags::empty(), None);
}
mir::StatementKind::FakeRead(..) |
mir::StatementKind::AscribeUserType(..) |
mir::StatementKind::ConstEvalCounter |
mir::StatementKind::PlaceMention(..) |
mir::StatementKind::BackwardIncompatibleDropHint { .. } |
mir::StatementKind::Nop => {}
}
}
}
}#[instrument(level = "debug", skip(self, bx))]
11 pub(crate) fn codegen_statement(&mut self, bx: &mut Bx, statement: &mir::Statement<'tcx>) {
12 self.codegen_stmt_debuginfos(bx, &statement.debuginfos);
13 self.set_debug_loc(bx, statement.source_info);
14 match statement.kind {
15 mir::StatementKind::Assign((ref place, ref rvalue)) => {
16 let needs_retag = bx.tcx().sess.opts.unstable_opts.codegen_emit_retag.is_some()
17 && retag::rvalue_needs_retag(rvalue);
18
19 if let Some(index) = place.as_local() {
20 match self.locals[index] {
21 LocalRef::Place(cg_dest) => {
22 self.codegen_rvalue(bx, cg_dest, rvalue);
23 if needs_retag {
24 self.codegen_retag_place(bx, cg_dest, false);
25 }
26 }
27 LocalRef::UnsizedPlace(cg_indirect_dest) => {
28 let ty = cg_indirect_dest.layout.ty;
29 span_bug!(
30 statement.source_info.span,
31 "cannot reallocate from `UnsizedPlace({ty})` \
32 into `{rvalue:?}`; dynamic alloca is not supported",
33 );
34 }
35 LocalRef::PendingOperand => {
36 let mut operand = self.codegen_rvalue_operand(bx, rvalue);
37 if needs_retag {
38 operand = self.codegen_retag_operand(bx, operand, false);
39 }
40 self.overwrite_local(index, LocalRef::Operand(operand));
41 self.debug_introduce_local(bx, index);
42 }
43 LocalRef::Operand(op) => {
44 if !op.layout.is_zst() {
45 span_bug!(
46 statement.source_info.span,
47 "operand {:?} already assigned",
48 rvalue
49 );
50 }
51
52 let operand = self.codegen_rvalue_operand(bx, rvalue);
56 if needs_retag {
57 self.codegen_retag_operand(bx, operand, false);
58 }
59 }
60 }
61 } else {
62 let cg_dest = self.codegen_place(bx, place.as_ref());
63 self.codegen_rvalue(bx, cg_dest, rvalue);
64 if needs_retag {
65 self.codegen_retag_place(bx, cg_dest, false);
66 }
67 }
68 }
69 mir::StatementKind::SetDiscriminant { ref place, variant_index } => {
70 self.codegen_place(bx, (**place).as_ref()).codegen_set_discr(bx, variant_index);
71 }
72 mir::StatementKind::StorageLive(local) => {
73 if let LocalRef::Place(cg_place) = self.locals[local] {
74 cg_place.storage_live(bx);
75 } else if let LocalRef::UnsizedPlace(cg_indirect_place) = self.locals[local] {
76 cg_indirect_place.storage_live(bx);
77 }
78 }
79 mir::StatementKind::StorageDead(local) => {
80 if let LocalRef::Place(cg_place) = self.locals[local] {
81 cg_place.storage_dead(bx);
82 } else if let LocalRef::UnsizedPlace(cg_indirect_place) = self.locals[local] {
83 cg_indirect_place.storage_dead(bx);
84 }
85 }
86 mir::StatementKind::Coverage(ref kind) => {
87 self.codegen_coverage(bx, kind, statement.source_info.scope);
88 }
89 mir::StatementKind::Intrinsic(NonDivergingIntrinsic::Assume(ref op)) => {
90 let op_val = self.codegen_operand(bx, op);
91 bx.assume(op_val.immediate());
92 }
93 mir::StatementKind::Intrinsic(NonDivergingIntrinsic::CopyNonOverlapping(
94 mir::CopyNonOverlapping { ref count, ref src, ref dst },
95 )) => {
96 let dst_val = self.codegen_operand(bx, dst);
97 let src_val = self.codegen_operand(bx, src);
98 let count = self.codegen_operand(bx, count).immediate();
99
100 let &ty::RawPtr(pointee, _) = dst_val.layout.ty.kind() else {
101 bug!("expected pointer")
102 };
103 let pointee_layout = bx
104 .tcx()
105 .layout_of(bx.typing_env().as_query_input(pointee))
106 .expect("expected pointee to have a layout");
107 let elem_size = pointee_layout.layout.size().bytes();
108 let bytes = bx.unchecked_sumul(count, bx.const_usize(elem_size));
109
110 let align = pointee_layout.layout.align.abi;
111 let dst = dst_val.immediate();
112 let src = src_val.immediate();
113
114 bx.memcpy(dst, align, src, align, bytes, crate::MemFlags::empty(), None);
115 }
116 mir::StatementKind::FakeRead(..)
117 | mir::StatementKind::AscribeUserType(..)
118 | mir::StatementKind::ConstEvalCounter
119 | mir::StatementKind::PlaceMention(..)
120 | mir::StatementKind::BackwardIncompatibleDropHint { .. }
121 | mir::StatementKind::Nop => {}
122 }
123 }
124
125 pub(crate) fn codegen_stmt_debuginfo(&mut self, bx: &mut Bx, debuginfo: &StmtDebugInfo<'tcx>) {
126 match debuginfo {
127 StmtDebugInfo::AssignRef(dest, place) => {
128 let local_ref = match self.locals[place.local] {
129 LocalRef::Place(place_ref) => Some((place_ref, place.projection.as_slice())),
132 LocalRef::Operand(operand_ref) if place.is_indirect_first_projection() => {
135 Some((operand_ref.deref(bx.cx()), &place.projection[1..]))
136 }
137 LocalRef::Operand(_) => None,
141 LocalRef::UnsizedPlace(_) | LocalRef::PendingOperand => None,
142 }
143 .filter(|(_, projection)| {
144 projection.iter().all(|p| p.can_use_in_debuginfo())
146 });
147 if let Some((base, projection)) = local_ref {
148 self.debug_new_val_to_local(bx, *dest, base, projection);
149 } else {
150 self.debug_poison_to_local(bx, *dest);
152 }
153 }
154 StmtDebugInfo::InvalidAssign(local) => {
155 self.debug_poison_to_local(bx, *local);
156 }
157 }
158 }
159
160 pub(crate) fn codegen_stmt_debuginfos(
161 &mut self,
162 bx: &mut Bx,
163 debuginfos: &[StmtDebugInfo<'tcx>],
164 ) {
165 for debuginfo in debuginfos {
166 self.codegen_stmt_debuginfo(bx, debuginfo);
167 }
168 }
169}