Skip to content

Commit 62d956f

Browse files
committed
Correctly change type when adding adjustments on top of NeverToAny
1 parent 1737162 commit 62d956f

File tree

4 files changed

+25
-41
lines changed

4 files changed

+25
-41
lines changed

compiler/rustc_hir_typeck/src/fn_ctxt/_impl.rs

+23-10
Original file line numberDiff line numberDiff line change
@@ -279,13 +279,23 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
279279
}
280280
Entry::Occupied(mut entry) => {
281281
debug!(" - composing on top of {:?}", entry.get());
282-
match (&entry.get()[..], &adj[..]) {
283-
// Applying any adjustment on top of a NeverToAny
284-
// is a valid NeverToAny adjustment, because it can't
285-
// be reached.
286-
(&[Adjustment { kind: Adjust::NeverToAny, .. }], _) => return,
282+
match (&mut entry.get_mut()[..], &adj[..]) {
287283
(
288-
&[
284+
[Adjustment { kind: Adjust::NeverToAny, target }],
285+
&[.., Adjustment { target: new_target, .. }],
286+
) => {
287+
// NeverToAny coercion can target any type, so instead of adding a new
288+
// adjustment on top we can change the target.
289+
//
290+
// This is required for things like `a == a` (where `a: !`) to produce
291+
// valid MIR -- we need borrow adjustment from things like `==` to change
292+
// the type to `&!` (or `&()` depending on the fallback). This might be
293+
// relevant even in unreachable code.
294+
*target = new_target;
295+
}
296+
297+
(
298+
&mut [
289299
Adjustment { kind: Adjust::Deref(_), .. },
290300
Adjustment { kind: Adjust::Borrow(AutoBorrow::Ref(..)), .. },
291301
],
@@ -294,11 +304,13 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
294304
.., // Any following adjustments are allowed.
295305
],
296306
) => {
297-
// A reborrow has no effect before a dereference.
307+
// A reborrow has no effect before a dereference, so we can safely replace adjustments.
308+
*entry.get_mut() = adj;
298309
}
299-
// FIXME: currently we never try to compose autoderefs
300-
// and ReifyFnPointer/UnsafeFnPointer, but we could.
310+
301311
_ => {
312+
// FIXME: currently we never try to compose autoderefs
313+
// and ReifyFnPointer/UnsafeFnPointer, but we could.
302314
self.dcx().span_delayed_bug(
303315
expr.span,
304316
format!(
@@ -308,9 +320,10 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
308320
adj
309321
),
310322
);
323+
324+
*entry.get_mut() = adj;
311325
}
312326
}
313-
*entry.get_mut() = adj;
314327
}
315328
}
316329

tests/mir-opt/building/eq_never_type._f.built.after.mir

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ fn _f(_1: !, _2: !) -> () {
66
let mut _0: ();
77
let mut _3: !;
88
let _4: bool;
9-
let mut _5: ();
9+
let mut _5: &();
1010
let mut _6: !;
1111
let mut _7: &();
1212
let _8: ();

tests/ui/never_type/eq-never-types.rs

+1-10
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,7 @@
1-
//@ known-bug: #120600
1+
//@ check-pass
22
//
33
// issue: rust-lang/rust#120600
44

5-
//@ failure-status: 101
6-
//@ normalize-stderr-test: "DefId\(.*?\]::" -> "DefId("
7-
//@ normalize-stderr-test: "(?m)note: we would appreciate a bug report.*\n\n" -> ""
8-
//@ normalize-stderr-test: "(?m)note: rustc.*running on.*\n\n" -> ""
9-
//@ normalize-stderr-test: "(?m)note: compiler flags.*\n\n" -> ""
10-
//@ normalize-stderr-test: "(?m)note: delayed at.*$" -> ""
11-
//@ normalize-stderr-test: "(?m)^ *\d+: .*\n" -> ""
12-
//@ normalize-stderr-test: "(?m)^ *at .*\n" -> ""
13-
145
#![allow(internal_features)]
156
#![feature(never_type, rustc_attrs)]
167
#![rustc_never_type_options(fallback = "never")]

tests/ui/never_type/eq-never-types.stderr

-20
This file was deleted.

0 commit comments

Comments
 (0)