]> git.proxmox.com Git - rustc.git/blame - compiler/rustc_borrowck/src/places_conflict.rs
New upstream version 1.67.1+dfsg1
[rustc.git] / compiler / rustc_borrowck / src / places_conflict.rs
CommitLineData
487cf647
FG
1#![deny(rustc::untranslatable_diagnostic)]
2#![deny(rustc::diagnostic_outside_of_impl)]
c295e0f8
XL
3use crate::ArtificialField;
4use crate::Overlap;
5use crate::{AccessDepth, Deep, Shallow};
dfeec247 6use rustc_hir as hir;
ba9703b0
XL
7use rustc_middle::mir::{Body, BorrowKind, Local, Place, PlaceElem, PlaceRef, ProjectionElem};
8use rustc_middle::ty::{self, TyCtxt};
8faf50e0 9use std::cmp::max;
cdc7bbd5 10use std::iter;
8faf50e0 11
0731742a
XL
12/// When checking if a place conflicts with another place, this enum is used to influence decisions
13/// where a place might be equal or disjoint with another place, such as if `a[i] == a[j]`.
14/// `PlaceConflictBias::Overlap` would bias toward assuming that `i` might equal `j` and that these
15/// places overlap. `PlaceConflictBias::NoOverlap` assumes that for the purposes of the predicate
16/// being run in the calling context, the conservative choice is to assume the compared indices
17/// are disjoint (and therefore, do not overlap).
18#[derive(Copy, Clone, Debug, Eq, PartialEq)]
923072b8 19pub(crate) enum PlaceConflictBias {
0731742a
XL
20 Overlap,
21 NoOverlap,
22}
23
24/// Helper function for checking if places conflict with a mutable borrow and deep access depth.
25/// This is used to check for places conflicting outside of the borrow checking code (such as in
26/// dataflow).
923072b8 27pub(crate) fn places_conflict<'tcx>(
dc9dc135
XL
28 tcx: TyCtxt<'tcx>,
29 body: &Body<'tcx>,
ba9703b0
XL
30 borrow_place: Place<'tcx>,
31 access_place: Place<'tcx>,
0731742a
XL
32 bias: PlaceConflictBias,
33) -> bool {
34 borrow_conflicts_with_place(
35 tcx,
dc9dc135 36 body,
0731742a
XL
37 borrow_place,
38 BorrowKind::Mut { allow_two_phase_borrow: true },
416331ca 39 access_place.as_ref(),
0731742a
XL
40 AccessDepth::Deep,
41 bias,
42 )
43}
44
45/// Checks whether the `borrow_place` conflicts with the `access_place` given a borrow kind and
46/// access depth. The `bias` parameter is used to determine how the unknowable (comparing runtime
47/// array indices, for example) should be interpreted - this depends on what the caller wants in
48/// order to make the conservative choice and preserve soundness.
f2b60f7d 49#[instrument(level = "debug", skip(tcx, body))]
dc9dc135
XL
50pub(super) fn borrow_conflicts_with_place<'tcx>(
51 tcx: TyCtxt<'tcx>,
52 body: &Body<'tcx>,
ba9703b0 53 borrow_place: Place<'tcx>,
0bf4aa26 54 borrow_kind: BorrowKind,
74b04a01 55 access_place: PlaceRef<'tcx>,
0bf4aa26 56 access: AccessDepth,
0731742a 57 bias: PlaceConflictBias,
8faf50e0 58) -> bool {
b7449926
XL
59 // This Local/Local case is handled by the more general code below, but
60 // it's so common that it's a speed win to check for it first.
5e7ed085
FG
61 if let Some(l1) = borrow_place.as_local() && let Some(l2) = access_place.as_local() {
62 return l1 == l2;
b7449926
XL
63 }
64
dfeec247 65 place_components_conflict(tcx, body, borrow_place, borrow_kind, access_place, access, bias)
8faf50e0
XL
66}
67
dc9dc135
XL
68fn place_components_conflict<'tcx>(
69 tcx: TyCtxt<'tcx>,
70 body: &Body<'tcx>,
ba9703b0 71 borrow_place: Place<'tcx>,
0bf4aa26 72 borrow_kind: BorrowKind,
74b04a01 73 access_place: PlaceRef<'tcx>,
0bf4aa26 74 access: AccessDepth,
0731742a 75 bias: PlaceConflictBias,
8faf50e0
XL
76) -> bool {
77 // The borrowck rules for proving disjointness are applied from the "root" of the
78 // borrow forwards, iterating over "similar" projections in lockstep until
79 // we can prove overlap one way or another. Essentially, we treat `Overlap` as
80 // a monoid and report a conflict if the product ends up not being `Disjoint`.
81 //
82 // At each step, if we didn't run out of borrow or place, we know that our elements
83 // have the same type, and that they only overlap if they are the identical.
84 //
85 // For example, if we are comparing these:
86 // BORROW: (*x1[2].y).z.a
87 // ACCESS: (*x1[i].y).w.b
88 //
89 // Then our steps are:
90 // x1 | x1 -- places are the same
91 // x1[2] | x1[i] -- equal or disjoint (disjoint if indexes differ)
92 // x1[2].y | x1[i].y -- equal or disjoint
93 // *x1[2].y | *x1[i].y -- equal or disjoint
94 // (*x1[2].y).z | (*x1[i].y).w -- we are disjoint and don't need to check more!
95 //
96 // Because `zip` does potentially bad things to the iterator inside, this loop
97 // also handles the case where the access might be a *prefix* of the borrow, e.g.
98 //
99 // BORROW: (*x1[2].y).z.a
100 // ACCESS: x1[i].y
101 //
102 // Then our steps are:
103 // x1 | x1 -- places are the same
104 // x1[2] | x1[i] -- equal or disjoint (disjoint if indexes differ)
105 // x1[2].y | x1[i].y -- equal or disjoint
106 //
107 // -- here we run out of access - the borrow can access a part of it. If this
108 // is a full deep access, then we *know* the borrow conflicts with it. However,
109 // if the access is shallow, then we can proceed:
110 //
111 // x1[2].y | (*x1[i].y) -- a deref! the access can't get past this, so we
112 // are disjoint
113 //
114 // Our invariant is, that at each step of the iteration:
115 // - If we didn't run out of access to match, our borrow and access are comparable
116 // and either equal or disjoint.
b7449926 117 // - If we did run out of access, the borrow can access a part of it.
48663c56 118
74b04a01 119 let borrow_local = borrow_place.local;
dfeec247 120 let access_local = access_place.local;
48663c56 121
dfeec247 122 match place_base_conflict(borrow_local, access_local) {
48663c56
XL
123 Overlap::Arbitrary => {
124 bug!("Two base can't return Arbitrary");
125 }
126 Overlap::EqualOrDisjoint => {
127 // This is the recursive case - proceed to the next element.
128 }
129 Overlap::Disjoint => {
130 // We have proven the borrow disjoint - further
131 // projections will remain disjoint.
132 debug!("borrow_conflicts_with_place: disjoint");
133 return false;
134 }
135 }
136
e1599b0c 137 // loop invariant: borrow_c is always either equal to access_c or disjoint from it.
f9f354fc 138 for (i, (borrow_c, &access_c)) in
cdc7bbd5 139 iter::zip(borrow_place.projection, access_place.projection).enumerate()
e1599b0c 140 {
f2b60f7d 141 debug!(?borrow_c, ?access_c);
8faf50e0 142
f2b60f7d 143 let borrow_proj_base = &borrow_place.projection[..i];
8faf50e0 144
e1599b0c
XL
145 // Borrow and access path both have more components.
146 //
147 // Examples:
148 //
149 // - borrow of `a.(...)`, access to `a.(...)`
150 // - borrow of `a.(...)`, access to `b.(...)`
151 //
152 // Here we only see the components we have checked so
153 // far (in our examples, just the first component). We
154 // check whether the components being borrowed vs
155 // accessed are disjoint (as in the second example,
156 // but not the first).
157 match place_projection_conflict(
158 tcx,
159 body,
dfeec247 160 borrow_local,
e1599b0c
XL
161 borrow_proj_base,
162 borrow_c,
163 access_c,
164 bias,
165 ) {
166 Overlap::Arbitrary => {
167 // We have encountered different fields of potentially
168 // the same union - the borrow now partially overlaps.
8faf50e0 169 //
e1599b0c
XL
170 // There is no *easy* way of comparing the fields
171 // further on, because they might have different types
172 // (e.g., borrows of `u.a.0` and `u.b.y` where `.0` and
173 // `.y` come from different structs).
8faf50e0 174 //
e1599b0c
XL
175 // We could try to do some things here - e.g., count
176 // dereferences - but that's probably not a good
177 // idea, at least for now, so just give up and
178 // report a conflict. This is unsafe code anyway so
179 // the user could always use raw pointers.
f2b60f7d 180 debug!("arbitrary -> conflict");
e1599b0c
XL
181 return true;
182 }
183 Overlap::EqualOrDisjoint => {
184 // This is the recursive case - proceed to the next element.
185 }
186 Overlap::Disjoint => {
187 // We have proven the borrow disjoint - further
188 // projections will remain disjoint.
f2b60f7d 189 debug!("disjoint");
e1599b0c
XL
190 return false;
191 }
192 }
193 }
194
195 if borrow_place.projection.len() > access_place.projection.len() {
196 for (i, elem) in borrow_place.projection[access_place.projection.len()..].iter().enumerate()
197 {
198 // Borrow path is longer than the access path. Examples:
199 //
200 // - borrow of `a.b.c`, access to `a.b`
201 //
202 // Here, we know that the borrow can access a part of
203 // our place. This is a conflict if that is a part our
204 // access cares about.
8faf50e0 205
e1599b0c 206 let proj_base = &borrow_place.projection[..access_place.projection.len() + i];
dfeec247 207 let base_ty = Place::ty_from(borrow_local, proj_base, body, tcx).ty;
8faf50e0 208
1b1a35ee 209 match (elem, &base_ty.kind(), access) {
e1599b0c
XL
210 (_, _, Shallow(Some(ArtificialField::ArrayLength)))
211 | (_, _, Shallow(Some(ArtificialField::ShallowBorrow))) => {
212 // The array length is like additional fields on the
213 // type; it does not overlap any existing data there.
214 // Furthermore, if cannot actually be a prefix of any
215 // borrowed place (at least in MIR as it is currently.)
216 //
217 // e.g., a (mutable) borrow of `a[5]` while we read the
218 // array length of `a`.
219 debug!("borrow_conflicts_with_place: implicit field");
220 return false;
221 }
8faf50e0 222
e1599b0c
XL
223 (ProjectionElem::Deref, _, Shallow(None)) => {
224 // e.g., a borrow of `*x.y` while we shallowly access `x.y` or some
225 // prefix thereof - the shallow access can't touch anything behind
226 // the pointer.
227 debug!("borrow_conflicts_with_place: shallow access behind ptr");
228 return false;
229 }
dfeec247 230 (ProjectionElem::Deref, ty::Ref(_, _, hir::Mutability::Not), _) => {
e1599b0c
XL
231 // Shouldn't be tracked
232 bug!("Tracking borrow behind shared reference.");
233 }
dfeec247 234 (ProjectionElem::Deref, ty::Ref(_, _, hir::Mutability::Mut), AccessDepth::Drop) => {
e1599b0c
XL
235 // Values behind a mutable reference are not access either by dropping a
236 // value, or by StorageDead
237 debug!("borrow_conflicts_with_place: drop access behind ptr");
238 return false;
239 }
8faf50e0 240
e1599b0c
XL
241 (ProjectionElem::Field { .. }, ty::Adt(def, _), AccessDepth::Drop) => {
242 // Drop can read/write arbitrary projections, so places
243 // conflict regardless of further projections.
244 if def.has_dtor(tcx) {
245 return true;
0bf4aa26 246 }
e1599b0c 247 }
0bf4aa26 248
e1599b0c
XL
249 (ProjectionElem::Deref, _, Deep)
250 | (ProjectionElem::Deref, _, AccessDepth::Drop)
251 | (ProjectionElem::Field { .. }, _, _)
252 | (ProjectionElem::Index { .. }, _, _)
253 | (ProjectionElem::ConstantIndex { .. }, _, _)
254 | (ProjectionElem::Subslice { .. }, _, _)
2b03887a 255 | (ProjectionElem::OpaqueCast { .. }, _, _)
e1599b0c
XL
256 | (ProjectionElem::Downcast { .. }, _, _) => {
257 // Recursive case. This can still be disjoint on a
258 // further iteration if this a shallow access and
259 // there's a deref later on, e.g., a borrow
260 // of `*x.y` while accessing `x`.
8faf50e0
XL
261 }
262 }
8faf50e0
XL
263 }
264 }
e1599b0c
XL
265
266 // Borrow path ran out but access path may not
267 // have. Examples:
268 //
269 // - borrow of `a.b`, access to `a.b.c`
270 // - borrow of `a.b`, access to `a.b`
271 //
272 // In the first example, where we didn't run out of
273 // access, the borrow can access all of our place, so we
274 // have a conflict.
275 //
276 // If the second example, where we did, then we still know
277 // that the borrow can access a *part* of our place that
278 // our access cares about, so we still have a conflict.
279 if borrow_kind == BorrowKind::Shallow
280 && borrow_place.projection.len() < access_place.projection.len()
281 {
282 debug!("borrow_conflicts_with_place: shallow borrow");
283 false
284 } else {
285 debug!("borrow_conflicts_with_place: full borrow, CONFLICT");
286 true
287 }
8faf50e0
XL
288}
289
8faf50e0
XL
290// Given that the bases of `elem1` and `elem2` are always either equal
291// or disjoint (and have the same type!), return the overlap situation
292// between `elem1` and `elem2`.
74b04a01 293fn place_base_conflict(l1: Local, l2: Local) -> Overlap {
dfeec247
XL
294 if l1 == l2 {
295 // the same local - base case, equal
296 debug!("place_element_conflict: DISJOINT-OR-EQ-LOCAL");
297 Overlap::EqualOrDisjoint
298 } else {
299 // different locals - base case, disjoint
300 debug!("place_element_conflict: DISJOINT-LOCAL");
301 Overlap::Disjoint
48663c56
XL
302 }
303}
304
305// Given that the bases of `elem1` and `elem2` are always either equal
306// or disjoint (and have the same type!), return the overlap situation
307// between `elem1` and `elem2`.
dc9dc135
XL
308fn place_projection_conflict<'tcx>(
309 tcx: TyCtxt<'tcx>,
310 body: &Body<'tcx>,
74b04a01 311 pi1_local: Local,
e1599b0c 312 pi1_proj_base: &[PlaceElem<'tcx>],
f9f354fc
XL
313 pi1_elem: PlaceElem<'tcx>,
314 pi2_elem: PlaceElem<'tcx>,
48663c56
XL
315 bias: PlaceConflictBias,
316) -> Overlap {
e1599b0c 317 match (pi1_elem, pi2_elem) {
48663c56
XL
318 (ProjectionElem::Deref, ProjectionElem::Deref) => {
319 // derefs (e.g., `*x` vs. `*x`) - recur.
320 debug!("place_element_conflict: DISJOINT-OR-EQ-DEREF");
321 Overlap::EqualOrDisjoint
322 }
487cf647
FG
323 (ProjectionElem::OpaqueCast(_), ProjectionElem::OpaqueCast(_)) => {
324 // casts to other types may always conflict irrespective of the type being cast to.
325 debug!("place_element_conflict: DISJOINT-OR-EQ-OPAQUE");
326 Overlap::EqualOrDisjoint
2b03887a 327 }
48663c56
XL
328 (ProjectionElem::Field(f1, _), ProjectionElem::Field(f2, _)) => {
329 if f1 == f2 {
330 // same field (e.g., `a.y` vs. `a.y`) - recur.
331 debug!("place_element_conflict: DISJOINT-OR-EQ-FIELD");
332 Overlap::EqualOrDisjoint
333 } else {
dfeec247 334 let ty = Place::ty_from(pi1_local, pi1_proj_base, body, tcx).ty;
17df50a5
XL
335 if ty.is_union() {
336 // Different fields of a union, we are basically stuck.
337 debug!("place_element_conflict: STUCK-UNION");
338 Overlap::Arbitrary
339 } else {
340 // Different fields of a struct (`a.x` vs. `a.y`). Disjoint!
341 debug!("place_element_conflict: DISJOINT-FIELD");
342 Overlap::Disjoint
8faf50e0 343 }
48663c56
XL
344 }
345 }
346 (ProjectionElem::Downcast(_, v1), ProjectionElem::Downcast(_, v2)) => {
347 // different variants are treated as having disjoint fields,
348 // even if they occupy the same "space", because it's
349 // impossible for 2 variants of the same enum to exist
350 // (and therefore, to be borrowed) at the same time.
351 //
352 // Note that this is different from unions - we *do* allow
353 // this code to compile:
354 //
355 // ```
356 // fn foo(x: &mut Result<i32, i32>) {
357 // let mut v = None;
358 // if let Ok(ref mut a) = *x {
359 // v = Some(a);
360 // }
361 // // here, you would *think* that the
362 // // *entirety* of `x` would be borrowed,
363 // // but in fact only the `Ok` variant is,
364 // // so the `Err` variant is *entirely free*:
365 // if let Err(ref mut a) = *x {
366 // v = Some(a);
367 // }
368 // drop(v);
369 // }
370 // ```
371 if v1 == v2 {
372 debug!("place_element_conflict: DISJOINT-OR-EQ-FIELD");
373 Overlap::EqualOrDisjoint
374 } else {
375 debug!("place_element_conflict: DISJOINT-FIELD");
376 Overlap::Disjoint
377 }
378 }
ba9703b0
XL
379 (
380 ProjectionElem::Index(..),
381 ProjectionElem::Index(..)
382 | ProjectionElem::ConstantIndex { .. }
383 | ProjectionElem::Subslice { .. },
384 )
385 | (
386 ProjectionElem::ConstantIndex { .. } | ProjectionElem::Subslice { .. },
387 ProjectionElem::Index(..),
388 ) => {
48663c56
XL
389 // Array indexes (`a[0]` vs. `a[i]`). These can either be disjoint
390 // (if the indexes differ) or equal (if they are the same).
391 match bias {
392 PlaceConflictBias::Overlap => {
393 // If we are biased towards overlapping, then this is the recursive
394 // case that gives "equal *or* disjoint" its meaning.
395 debug!("place_element_conflict: DISJOINT-OR-EQ-ARRAY-INDEX");
396 Overlap::EqualOrDisjoint
8faf50e0 397 }
48663c56
XL
398 PlaceConflictBias::NoOverlap => {
399 // If we are biased towards no overlapping, then this is disjoint.
400 debug!("place_element_conflict: DISJOINT-ARRAY-INDEX");
401 Overlap::Disjoint
8faf50e0 402 }
8faf50e0
XL
403 }
404 }
dfeec247
XL
405 (
406 ProjectionElem::ConstantIndex { offset: o1, min_length: _, from_end: false },
407 ProjectionElem::ConstantIndex { offset: o2, min_length: _, from_end: false },
408 )
409 | (
410 ProjectionElem::ConstantIndex { offset: o1, min_length: _, from_end: true },
411 ProjectionElem::ConstantIndex { offset: o2, min_length: _, from_end: true },
412 ) => {
48663c56
XL
413 if o1 == o2 {
414 debug!("place_element_conflict: DISJOINT-OR-EQ-ARRAY-CONSTANT-INDEX");
415 Overlap::EqualOrDisjoint
416 } else {
417 debug!("place_element_conflict: DISJOINT-ARRAY-CONSTANT-INDEX");
418 Overlap::Disjoint
419 }
420 }
dfeec247 421 (
48663c56 422 ProjectionElem::ConstantIndex {
dfeec247
XL
423 offset: offset_from_begin,
424 min_length: min_length1,
425 from_end: false,
426 },
427 ProjectionElem::ConstantIndex {
428 offset: offset_from_end,
429 min_length: min_length2,
430 from_end: true,
431 },
432 )
433 | (
434 ProjectionElem::ConstantIndex {
435 offset: offset_from_end,
436 min_length: min_length1,
437 from_end: true,
438 },
439 ProjectionElem::ConstantIndex {
440 offset: offset_from_begin,
441 min_length: min_length2,
442 from_end: false,
443 },
444 ) => {
48663c56
XL
445 // both patterns matched so it must be at least the greater of the two
446 let min_length = max(min_length1, min_length2);
447 // `offset_from_end` can be in range `[1..min_length]`, 1 indicates the last
448 // element (like -1 in Python) and `min_length` the first.
449 // Therefore, `min_length - offset_from_end` gives the minimal possible
450 // offset from the beginning
f9f354fc 451 if offset_from_begin >= min_length - offset_from_end {
48663c56
XL
452 debug!("place_element_conflict: DISJOINT-OR-EQ-ARRAY-CONSTANT-INDEX-FE");
453 Overlap::EqualOrDisjoint
454 } else {
455 debug!("place_element_conflict: DISJOINT-ARRAY-CONSTANT-INDEX-FE");
456 Overlap::Disjoint
457 }
458 }
60c5eb7d
XL
459 (
460 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: false },
dfeec247 461 ProjectionElem::Subslice { from, to, from_end: false },
60c5eb7d
XL
462 )
463 | (
464 ProjectionElem::Subslice { from, to, from_end: false },
dfeec247 465 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: false },
60c5eb7d
XL
466 ) => {
467 if (from..to).contains(&offset) {
468 debug!("place_element_conflict: DISJOINT-OR-EQ-ARRAY-CONSTANT-INDEX-SUBSLICE");
469 Overlap::EqualOrDisjoint
470 } else {
471 debug!("place_element_conflict: DISJOINT-ARRAY-CONSTANT-INDEX-SUBSLICE");
472 Overlap::Disjoint
473 }
474 }
dfeec247
XL
475 (
476 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: false },
477 ProjectionElem::Subslice { from, .. },
478 )
479 | (
480 ProjectionElem::Subslice { from, .. },
481 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: false },
482 ) => {
48663c56 483 if offset >= from {
dfeec247 484 debug!("place_element_conflict: DISJOINT-OR-EQ-SLICE-CONSTANT-INDEX-SUBSLICE");
48663c56
XL
485 Overlap::EqualOrDisjoint
486 } else {
60c5eb7d 487 debug!("place_element_conflict: DISJOINT-SLICE-CONSTANT-INDEX-SUBSLICE");
48663c56
XL
488 Overlap::Disjoint
489 }
490 }
dfeec247
XL
491 (
492 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: true },
493 ProjectionElem::Subslice { to, from_end: true, .. },
494 )
495 | (
496 ProjectionElem::Subslice { to, from_end: true, .. },
497 ProjectionElem::ConstantIndex { offset, min_length: _, from_end: true },
498 ) => {
48663c56 499 if offset > to {
dfeec247
XL
500 debug!(
501 "place_element_conflict: \
502 DISJOINT-OR-EQ-SLICE-CONSTANT-INDEX-SUBSLICE-FE"
503 );
48663c56
XL
504 Overlap::EqualOrDisjoint
505 } else {
60c5eb7d
XL
506 debug!("place_element_conflict: DISJOINT-SLICE-CONSTANT-INDEX-SUBSLICE-FE");
507 Overlap::Disjoint
508 }
509 }
510 (
511 ProjectionElem::Subslice { from: f1, to: t1, from_end: false },
dfeec247 512 ProjectionElem::Subslice { from: f2, to: t2, from_end: false },
60c5eb7d
XL
513 ) => {
514 if f2 >= t1 || f1 >= t2 {
515 debug!("place_element_conflict: DISJOINT-ARRAY-SUBSLICES");
48663c56 516 Overlap::Disjoint
60c5eb7d
XL
517 } else {
518 debug!("place_element_conflict: DISJOINT-OR-EQ-ARRAY-SUBSLICES");
519 Overlap::EqualOrDisjoint
48663c56
XL
520 }
521 }
522 (ProjectionElem::Subslice { .. }, ProjectionElem::Subslice { .. }) => {
60c5eb7d 523 debug!("place_element_conflict: DISJOINT-OR-EQ-SLICE-SUBSLICES");
dfeec247 524 Overlap::EqualOrDisjoint
48663c56 525 }
ba9703b0
XL
526 (
527 ProjectionElem::Deref
528 | ProjectionElem::Field(..)
529 | ProjectionElem::Index(..)
530 | ProjectionElem::ConstantIndex { .. }
2b03887a 531 | ProjectionElem::OpaqueCast { .. }
ba9703b0
XL
532 | ProjectionElem::Subslice { .. }
533 | ProjectionElem::Downcast(..),
534 _,
535 ) => bug!(
48663c56 536 "mismatched projections in place_element_conflict: {:?} and {:?}",
e1599b0c
XL
537 pi1_elem,
538 pi2_elem
8faf50e0
XL
539 ),
540 }
541}