rustc_mir_build/builder/matches/
match_pair.rs1use std::sync::Arc;
2
3use rustc_abi::FieldIdx;
4use rustc_middle::mir::*;
5use rustc_middle::span_bug;
6use rustc_middle::thir::*;
7use rustc_middle::ty::{self, Ty, TypeVisitableExt};
8
9use crate::builder::Builder;
10use crate::builder::expr::as_place::{PlaceBase, PlaceBuilder};
11use crate::builder::matches::{
12 FlatPat, MatchPairTree, PatConstKind, PatternExtraData, SliceLenOp, TestableCase,
13};
14
15impl<'a, 'tcx> Builder<'a, 'tcx> {
16 fn field_match_pairs(
22 &mut self,
23 match_pairs: &mut Vec<MatchPairTree<'tcx>>,
24 extra_data: &mut PatternExtraData<'tcx>,
25 place: PlaceBuilder<'tcx>,
26 subpatterns: &[FieldPat<'tcx>],
27 ) {
28 for fieldpat in subpatterns {
29 let place = place.clone_project(PlaceElem::Field(fieldpat.field, fieldpat.pattern.ty));
30 MatchPairTree::for_pattern(place, &fieldpat.pattern, self, match_pairs, extra_data);
31 }
32 }
33
34 fn prefix_slice_suffix(
39 &mut self,
40 match_pairs: &mut Vec<MatchPairTree<'tcx>>,
41 extra_data: &mut PatternExtraData<'tcx>,
42 place: &PlaceBuilder<'tcx>,
43 array_len: Option<u64>,
44 prefix: &[Pat<'tcx>],
45 opt_slice: &Option<Box<Pat<'tcx>>>,
46 suffix: &[Pat<'tcx>],
47 ) {
48 let prefix_len = u64::try_from(prefix.len()).unwrap();
49 let suffix_len = u64::try_from(suffix.len()).unwrap();
50
51 let (min_length, is_array) = match array_len {
57 Some(len) => (len, true),
58 None => (prefix_len + suffix_len, false),
59 };
60
61 for (offset, subpattern) in (0u64..).zip(prefix) {
62 let elem = ProjectionElem::ConstantIndex { offset, min_length, from_end: false };
63 let place = place.clone_project(elem);
64 MatchPairTree::for_pattern(place, subpattern, self, match_pairs, extra_data)
65 }
66
67 if let Some(subslice_pat) = opt_slice {
68 let subslice = place.clone_project(PlaceElem::Subslice {
69 from: prefix_len,
70 to: if is_array { min_length - suffix_len } else { suffix_len },
71 from_end: !is_array,
72 });
73 MatchPairTree::for_pattern(subslice, subslice_pat, self, match_pairs, extra_data);
74 }
75
76 for (end_offset, subpattern) in (1u64..).zip(suffix.iter().rev()) {
77 let elem = ProjectionElem::ConstantIndex {
78 offset: if is_array { min_length - end_offset } else { end_offset },
79 min_length,
80 from_end: !is_array,
81 };
82 let place = place.clone_project(elem);
83 MatchPairTree::for_pattern(place, subpattern, self, match_pairs, extra_data)
84 }
85 }
86}
87
88impl<'tcx> MatchPairTree<'tcx> {
89 pub(super) fn for_pattern(
92 mut place_builder: PlaceBuilder<'tcx>,
93 pattern: &Pat<'tcx>,
94 cx: &mut Builder<'_, 'tcx>,
95 match_pairs: &mut Vec<Self>, extra_data: &mut PatternExtraData<'tcx>, ) {
98 if let Some(resolved) = place_builder.resolve_upvar(cx) {
101 place_builder = resolved;
102 }
103
104 if !cx.tcx.next_trait_solver_globally() {
105 let may_need_cast = match place_builder.base() {
108 PlaceBase::Local(local) => {
109 let ty =
110 Place::ty_from(local, place_builder.projection(), &cx.local_decls, cx.tcx)
111 .ty;
112 ty != pattern.ty && ty.has_opaque_types()
113 }
114 _ => true,
115 };
116 if may_need_cast {
117 place_builder = place_builder.project(ProjectionElem::OpaqueCast(pattern.ty));
118 }
119 }
120
121 let place = place_builder.try_to_place(cx);
122
123 if let Some(place) = place
125 && let Some(extra) = &pattern.extra
126 {
127 for &Ascription { ref annotation, variance } in &extra.ascriptions {
128 extra_data.ascriptions.push(super::Ascription {
129 source: place,
130 annotation: annotation.clone(),
131 variance,
132 });
133 }
134 }
135
136 let mut subpairs = Vec::new();
137 let testable_case = match pattern.kind {
138 PatKind::Missing | PatKind::Wild | PatKind::Error(_) => None,
139
140 PatKind::Or { ref pats } => {
141 let pats: Box<[FlatPat<'tcx>]> =
142 pats.iter().map(|pat| FlatPat::new(place_builder.clone(), pat, cx)).collect();
143 if !pats[0].extra_data.bindings.is_empty() {
144 extra_data.bindings.push(super::SubpatternBindings::FromOrPattern);
150 }
151 Some(TestableCase::Or { pats })
152 }
153
154 PatKind::Range(ref range) => {
155 if range.is_full_range(cx.tcx) == Some(true) {
156 None
157 } else {
158 Some(TestableCase::Range(Arc::clone(range)))
159 }
160 }
161
162 PatKind::Constant { value } => {
163 match (&pattern.ty, &value.ty) {
(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);
}
}
};assert_eq!(pattern.ty, value.ty);
164
165 let pat_ty = pattern.ty;
168 let const_kind = if pat_ty.is_bool() {
169 PatConstKind::Bool
170 } else if pat_ty.is_integral() || pat_ty.is_char() {
171 PatConstKind::IntOrChar
172 } else if pat_ty.is_floating_point() {
173 PatConstKind::Float
174 } else if pat_ty.is_str() {
175 PatConstKind::String
176 } else {
177 PatConstKind::Other
181 };
182 Some(TestableCase::Constant { value, kind: const_kind })
183 }
184
185 PatKind::Binding { mode, var, is_shorthand, ref subpattern, .. } => {
186 if let Some(subpattern) = subpattern.as_ref() {
207 MatchPairTree::for_pattern(
209 place_builder,
210 subpattern,
211 cx,
212 &mut subpairs,
213 extra_data,
214 );
215 }
216
217 if let Some(source) = place {
219 extra_data.bindings.push(super::SubpatternBindings::One(super::Binding {
220 span: pattern.span,
221 source,
222 var_id: var,
223 binding_mode: mode,
224 is_shorthand,
225 }));
226 }
227
228 None
229 }
230
231 PatKind::Array { ref prefix, ref slice, ref suffix } => {
232 let array_len = match pattern.ty.kind() {
237 ty::Array(_, len) => len.try_to_target_usize(cx.tcx),
238 _ => None,
239 };
240 if let Some(array_len) = array_len {
241 cx.prefix_slice_suffix(
242 &mut subpairs,
243 extra_data,
244 &place_builder,
245 Some(array_len),
246 prefix,
247 slice,
248 suffix,
249 );
250 } else {
251 cx.tcx.dcx().span_delayed_bug(
254 pattern.span,
255 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("array length in pattern couldn\'t be determined for ty={0:?}",
pattern.ty))
})format!(
256 "array length in pattern couldn't be determined for ty={:?}",
257 pattern.ty
258 ),
259 );
260 }
261
262 None
263 }
264 PatKind::Slice { ref prefix, ref slice, ref suffix } => {
265 cx.prefix_slice_suffix(
266 &mut subpairs,
267 extra_data,
268 &place_builder,
269 None,
270 prefix,
271 slice,
272 suffix,
273 );
274
275 if prefix.is_empty() && slice.is_some() && suffix.is_empty() {
276 None
279 } else {
280 Some(TestableCase::Slice {
284 len: u64::try_from(prefix.len() + suffix.len()).unwrap(),
285 op: if slice.is_some() {
286 SliceLenOp::GreaterOrEqual
287 } else {
288 SliceLenOp::Equal
289 },
290 })
291 }
292 }
293
294 PatKind::Variant { adt_def, variant_index, args, ref subpatterns } => {
295 let downcast_place = place_builder.downcast(adt_def, variant_index); cx.field_match_pairs(&mut subpairs, extra_data, downcast_place, subpatterns);
297
298 let irrefutable = adt_def.variants().iter_enumerated().all(|(i, v)| {
299 i == variant_index
300 || !v.inhabited_predicate(cx.tcx, adt_def).instantiate(cx.tcx, args).apply(
301 cx.tcx,
302 cx.infcx.typing_env(cx.param_env),
303 cx.def_id.into(),
304 )
305 }) && !adt_def.variant_list_has_applicable_non_exhaustive();
306 if irrefutable {
307 None
308 } else {
309 Some(TestableCase::Variant { adt_def, variant_index })
310 }
311 }
312
313 PatKind::Leaf { ref subpatterns } => {
314 cx.field_match_pairs(&mut subpairs, extra_data, place_builder, subpatterns);
315 None
316 }
317
318 PatKind::Deref { pin: Pinnedness::Pinned, ref subpattern } => {
319 let pinned_ref_ty = match pattern.ty.pinned_ty() {
320 Some(p_ty) if p_ty.is_ref() => p_ty,
321 _ => ::rustc_middle::util::bug::span_bug_fmt(pattern.span,
format_args!("bad type for pinned deref: {0:?}", pattern.ty))span_bug!(pattern.span, "bad type for pinned deref: {:?}", pattern.ty),
322 };
323 MatchPairTree::for_pattern(
324 place_builder.field(FieldIdx::ZERO, pinned_ref_ty).deref(),
326 subpattern,
327 cx,
328 &mut subpairs,
329 extra_data,
330 );
331
332 None
333 }
334
335 PatKind::Deref { pin: Pinnedness::Not, ref subpattern }
336 | PatKind::DerefPattern { ref subpattern, borrow: DerefPatBorrowMode::Box } => {
337 MatchPairTree::for_pattern(
338 place_builder.deref(),
339 subpattern,
340 cx,
341 &mut subpairs,
342 extra_data,
343 );
344 None
345 }
346
347 PatKind::DerefPattern {
348 ref subpattern,
349 borrow: DerefPatBorrowMode::Borrow(mutability),
350 } => {
351 let temp = cx.temp(
354 Ty::new_ref(cx.tcx, cx.tcx.lifetimes.re_erased, subpattern.ty, mutability),
355 pattern.span,
356 );
357 MatchPairTree::for_pattern(
358 PlaceBuilder::from(temp).deref(),
359 subpattern,
360 cx,
361 &mut subpairs,
362 extra_data,
363 );
364 Some(TestableCase::Deref { temp, mutability })
365 }
366
367 PatKind::Never => Some(TestableCase::Never),
368 };
369
370 if let Some(testable_case) = testable_case {
371 match_pairs.push(MatchPairTree {
379 place,
380 testable_case,
381 subpairs,
382 pattern_ty: pattern.ty,
383 pattern_span: pattern.span,
384 })
385 } else {
386 match_pairs.extend(subpairs);
389 }
390 }
391}