rustc_mir_dataflow/impls/
liveness.rs1use rustc_index::bit_set::DenseBitSet;
2use rustc_middle::mir::visit::{MutatingUseContext, NonMutatingUseContext, PlaceContext, Visitor};
3use rustc_middle::mir::{
4 self, CallReturnPlaces, Local, Location, Place, StatementKind, TerminatorEdges,
5};
6
7use crate::{Analysis, Backward, GenKill};
8
9pub struct MaybeLiveLocals;
27
28impl<'tcx> Analysis<'tcx> for MaybeLiveLocals {
29 type Domain = DenseBitSet<Local>;
30 type Direction = Backward;
31
32 const NAME: &'static str = "liveness";
33
34 fn bottom_value(&self, body: &mir::Body<'tcx>) -> Self::Domain {
35 DenseBitSet::new_empty(body.local_decls.len())
37 }
38
39 fn initialize_start_block(&self, _: &mir::Body<'tcx>, _: &mut Self::Domain) {
40 }
42
43 fn apply_primary_statement_effect(
44 &mut self,
45 state: &mut Self::Domain,
46 statement: &mir::Statement<'tcx>,
47 location: Location,
48 ) {
49 TransferFunction(state).visit_statement(statement, location);
50 }
51
52 fn apply_primary_terminator_effect<'mir>(
53 &mut self,
54 state: &mut Self::Domain,
55 terminator: &'mir mir::Terminator<'tcx>,
56 location: Location,
57 ) -> TerminatorEdges<'mir, 'tcx> {
58 TransferFunction(state).visit_terminator(terminator, location);
59 terminator.edges()
60 }
61
62 fn apply_call_return_effect(
63 &mut self,
64 state: &mut Self::Domain,
65 _block: mir::BasicBlock,
66 return_places: CallReturnPlaces<'_, 'tcx>,
67 ) {
68 if let CallReturnPlaces::Yield(resume_place) = return_places {
69 YieldResumeEffect(state).visit_place(
70 &resume_place,
71 PlaceContext::MutatingUse(MutatingUseContext::Yield),
72 Location::START,
73 )
74 } else {
75 return_places.for_each(|place| {
76 if let Some(local) = place.as_local() {
77 state.kill(local);
78 }
79 });
80 }
81 }
82}
83
84pub struct TransferFunction<'a>(pub &'a mut DenseBitSet<Local>);
85
86impl<'tcx> Visitor<'tcx> for TransferFunction<'_> {
87 fn visit_place(&mut self, place: &mir::Place<'tcx>, context: PlaceContext, location: Location) {
88 if let PlaceContext::MutatingUse(MutatingUseContext::Yield) = context {
89 return;
92 }
93
94 match DefUse::for_place(*place, context) {
95 DefUse::Def => {
96 if let PlaceContext::MutatingUse(
97 MutatingUseContext::Call | MutatingUseContext::AsmOutput,
98 ) = context
99 {
100 } else {
105 self.0.kill(place.local);
106 }
107 }
108 DefUse::Use => self.0.gen_(place.local),
109 DefUse::PartialWrite | DefUse::NonUse => {}
110 }
111
112 self.visit_projection(place.as_ref(), context, location);
113 }
114
115 fn visit_local(&mut self, local: Local, context: PlaceContext, _: Location) {
116 DefUse::apply(self.0, local.into(), context);
117 }
118}
119
120struct YieldResumeEffect<'a>(&'a mut DenseBitSet<Local>);
121
122impl<'tcx> Visitor<'tcx> for YieldResumeEffect<'_> {
123 fn visit_place(&mut self, place: &mir::Place<'tcx>, context: PlaceContext, location: Location) {
124 DefUse::apply(self.0, *place, context);
125 self.visit_projection(place.as_ref(), context, location);
126 }
127
128 fn visit_local(&mut self, local: Local, context: PlaceContext, _: Location) {
129 DefUse::apply(self.0, local.into(), context);
130 }
131}
132
133#[derive(Eq, PartialEq, Clone)]
134pub enum DefUse {
135 Def,
137 Use,
139 PartialWrite,
141 NonUse,
143}
144
145impl DefUse {
146 fn apply(state: &mut DenseBitSet<Local>, place: Place<'_>, context: PlaceContext) {
147 match DefUse::for_place(place, context) {
148 DefUse::Def => state.kill(place.local),
149 DefUse::Use => state.gen_(place.local),
150 DefUse::PartialWrite | DefUse::NonUse => {}
151 }
152 }
153
154 pub fn for_place(place: Place<'_>, context: PlaceContext) -> DefUse {
155 match context {
156 PlaceContext::NonUse(_) => DefUse::NonUse,
157
158 PlaceContext::MutatingUse(
159 MutatingUseContext::Call
160 | MutatingUseContext::Yield
161 | MutatingUseContext::AsmOutput
162 | MutatingUseContext::Store,
163 ) => {
164 if place.is_indirect() {
166 DefUse::Use
167 } else if place.projection.is_empty() {
168 DefUse::Def
169 } else {
170 DefUse::PartialWrite
171 }
172 }
173
174 PlaceContext::MutatingUse(MutatingUseContext::SetDiscriminant) => {
177 if place.is_indirect() { DefUse::Use } else { DefUse::PartialWrite }
178 }
179
180 PlaceContext::MutatingUse(
182 MutatingUseContext::RawBorrow
183 | MutatingUseContext::Borrow
184 | MutatingUseContext::Drop
185 | MutatingUseContext::Retag,
186 )
187 | PlaceContext::NonMutatingUse(
188 NonMutatingUseContext::RawBorrow
189 | NonMutatingUseContext::Copy
190 | NonMutatingUseContext::Inspect
191 | NonMutatingUseContext::Move
192 | NonMutatingUseContext::PlaceMention
193 | NonMutatingUseContext::FakeBorrow
194 | NonMutatingUseContext::SharedBorrow,
195 ) => DefUse::Use,
196
197 PlaceContext::MutatingUse(MutatingUseContext::Projection)
198 | PlaceContext::NonMutatingUse(NonMutatingUseContext::Projection) => {
199 unreachable!("A projection could be a def or a use and must be handled separately")
200 }
201 }
202 }
203}
204
205pub struct MaybeTransitiveLiveLocals<'a> {
211 always_live: &'a DenseBitSet<Local>,
212 debuginfo_locals: &'a DenseBitSet<Local>,
213}
214
215impl<'a> MaybeTransitiveLiveLocals<'a> {
216 pub fn new(
221 always_live: &'a DenseBitSet<Local>,
222 debuginfo_locals: &'a DenseBitSet<Local>,
223 ) -> Self {
224 MaybeTransitiveLiveLocals { always_live, debuginfo_locals }
225 }
226
227 pub fn can_be_removed_if_dead<'tcx>(
228 stmt_kind: &StatementKind<'tcx>,
229 always_live: &DenseBitSet<Local>,
230 debuginfo_locals: &'a DenseBitSet<Local>,
231 ) -> Option<Place<'tcx>> {
232 let destination = match stmt_kind {
234 StatementKind::Assign(box (place, rvalue)) => (rvalue.is_safe_to_remove()
235 && (!debuginfo_locals.contains(place.local)
238 || (place.as_local().is_some() && stmt_kind.as_debuginfo().is_some())))
239 .then_some(*place),
240 StatementKind::SetDiscriminant { place, .. } => {
241 (!debuginfo_locals.contains(place.local)).then_some(**place)
242 }
243 StatementKind::FakeRead(_)
244 | StatementKind::StorageLive(_)
245 | StatementKind::StorageDead(_)
246 | StatementKind::Retag(..)
247 | StatementKind::AscribeUserType(..)
248 | StatementKind::PlaceMention(..)
249 | StatementKind::Coverage(..)
250 | StatementKind::Intrinsic(..)
251 | StatementKind::ConstEvalCounter
252 | StatementKind::BackwardIncompatibleDropHint { .. }
253 | StatementKind::Nop => None,
254 };
255 if let Some(destination) = destination
256 && !destination.is_indirect()
257 && !always_live.contains(destination.local)
258 {
259 return Some(destination);
260 }
261 None
262 }
263}
264
265impl<'a, 'tcx> Analysis<'tcx> for MaybeTransitiveLiveLocals<'a> {
266 type Domain = DenseBitSet<Local>;
267 type Direction = Backward;
268
269 const NAME: &'static str = "transitive liveness";
270
271 fn bottom_value(&self, body: &mir::Body<'tcx>) -> Self::Domain {
272 DenseBitSet::new_empty(body.local_decls.len())
274 }
275
276 fn initialize_start_block(&self, _: &mir::Body<'tcx>, _: &mut Self::Domain) {
277 }
279
280 fn apply_primary_statement_effect(
281 &mut self,
282 state: &mut Self::Domain,
283 statement: &mir::Statement<'tcx>,
284 location: Location,
285 ) {
286 if let Some(destination) =
287 Self::can_be_removed_if_dead(&statement.kind, &self.always_live, &self.debuginfo_locals)
288 && !state.contains(destination.local)
289 {
290 return;
292 }
293 TransferFunction(state).visit_statement(statement, location);
294 }
295
296 fn apply_primary_terminator_effect<'mir>(
297 &mut self,
298 state: &mut Self::Domain,
299 terminator: &'mir mir::Terminator<'tcx>,
300 location: Location,
301 ) -> TerminatorEdges<'mir, 'tcx> {
302 TransferFunction(state).visit_terminator(terminator, location);
303 terminator.edges()
304 }
305
306 fn apply_call_return_effect(
307 &mut self,
308 state: &mut Self::Domain,
309 _block: mir::BasicBlock,
310 return_places: CallReturnPlaces<'_, 'tcx>,
311 ) {
312 if let CallReturnPlaces::Yield(resume_place) = return_places {
313 YieldResumeEffect(state).visit_place(
314 &resume_place,
315 PlaceContext::MutatingUse(MutatingUseContext::Yield),
316 Location::START,
317 )
318 } else {
319 return_places.for_each(|place| {
320 if let Some(local) = place.as_local() {
321 state.remove(local);
322 }
323 });
324 }
325 }
326}