Rollup merge of #144216 - Nadrieril:revert-pin-hack, r=compiler-errors
Don't consider unstable fields always-inhabited This reverts the hack in https://github.com/rust-lang/rust/pull/133889 now that `Pin`'s field is no longer public. Fixes https://github.com/rust-lang/rust/issues/143468. r? ```@compiler-errors```
This commit is contained in:
commit
830b8237de
7 changed files with 71 additions and 72 deletions
|
|
@ -43,7 +43,6 @@
|
|||
//! This code should only compile in modules where the uninhabitedness of `Foo`
|
||||
//! is visible.
|
||||
|
||||
use rustc_span::sym;
|
||||
use rustc_type_ir::TyKind::*;
|
||||
use tracing::instrument;
|
||||
|
||||
|
|
@ -85,21 +84,6 @@ impl<'tcx> VariantDef {
|
|||
InhabitedPredicate::all(
|
||||
tcx,
|
||||
self.fields.iter().map(|field| {
|
||||
// Unstable fields are always considered to be inhabited. In the future,
|
||||
// this could be extended to be conditional on the field being unstable
|
||||
// only within the module that's querying the inhabitedness, like:
|
||||
// `let pred = pred.or(InhabitedPredicate::IsUnstable(field.did));`
|
||||
// but this is unnecessary for now, since it would only affect nightly-only
|
||||
// code or code within the standard library itself.
|
||||
// HACK: We filter out `rustc_private` fields since with the flag
|
||||
// `-Zforce-unstable-if-unmarked` we consider all unmarked fields to be
|
||||
// unstable when building the compiler.
|
||||
if tcx
|
||||
.lookup_stability(field.did)
|
||||
.is_some_and(|stab| stab.is_unstable() && stab.feature != sym::rustc_private)
|
||||
{
|
||||
return InhabitedPredicate::True;
|
||||
}
|
||||
let pred = tcx.type_of(field.did).instantiate_identity().inhabited_predicate(tcx);
|
||||
if adt.is_enum() {
|
||||
return pred;
|
||||
|
|
|
|||
|
|
@ -16,7 +16,7 @@ use rustc_middle::ty::{
|
|||
};
|
||||
use rustc_middle::{bug, span_bug};
|
||||
use rustc_session::lint;
|
||||
use rustc_span::{DUMMY_SP, ErrorGuaranteed, Span, sym};
|
||||
use rustc_span::{DUMMY_SP, ErrorGuaranteed, Span};
|
||||
|
||||
use crate::constructor::Constructor::*;
|
||||
use crate::constructor::{
|
||||
|
|
@ -238,10 +238,7 @@ impl<'p, 'tcx: 'p> RustcPatCtxt<'p, 'tcx> {
|
|||
let is_visible =
|
||||
adt.is_enum() || field.vis.is_accessible_from(cx.module, cx.tcx);
|
||||
let is_uninhabited = cx.is_uninhabited(*ty);
|
||||
let is_unstable = cx.tcx.lookup_stability(field.did).is_some_and(|stab| {
|
||||
stab.is_unstable() && stab.feature != sym::rustc_private
|
||||
});
|
||||
let skip = is_uninhabited && (!is_visible || is_unstable);
|
||||
let skip = is_uninhabited && !is_visible;
|
||||
(ty, PrivateUninhabitedField(skip))
|
||||
});
|
||||
cx.dropless_arena.alloc_from_iter(tys)
|
||||
|
|
|
|||
|
|
@ -6,3 +6,9 @@ pub struct Foo<T> {
|
|||
#[unstable(feature = "unstable", issue = "none")]
|
||||
pub field: T,
|
||||
}
|
||||
|
||||
#[unstable(feature = "my_coro_state", issue = "none")]
|
||||
pub enum MyCoroutineState<Y, R> {
|
||||
Yielded(Y),
|
||||
Complete(R),
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,22 +0,0 @@
|
|||
error[E0004]: non-exhaustive patterns: type `Foo<Void>` is non-empty
|
||||
--> $DIR/uninhabited-unstable-field.rs:13:11
|
||||
|
|
||||
LL | match x {}
|
||||
| ^
|
||||
|
|
||||
note: `Foo<Void>` defined here
|
||||
--> $DIR/auxiliary/staged-api.rs:5:1
|
||||
|
|
||||
LL | pub struct Foo<T> {
|
||||
| ^^^^^^^^^^^^^^^^^
|
||||
= note: the matched value is of type `Foo<Void>`
|
||||
help: ensure that all possible cases are being handled by adding a match arm with a wildcard pattern as shown
|
||||
|
|
||||
LL ~ match x {
|
||||
LL + _ => todo!(),
|
||||
LL ~ }
|
||||
|
|
||||
|
||||
error: aborting due to 1 previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0004`.
|
||||
|
|
@ -1,22 +0,0 @@
|
|||
error[E0004]: non-exhaustive patterns: type `Foo<Void>` is non-empty
|
||||
--> $DIR/uninhabited-unstable-field.rs:13:11
|
||||
|
|
||||
LL | match x {}
|
||||
| ^
|
||||
|
|
||||
note: `Foo<Void>` defined here
|
||||
--> $DIR/auxiliary/staged-api.rs:5:1
|
||||
|
|
||||
LL | pub struct Foo<T> {
|
||||
| ^^^^^^^^^^^^^^^^^
|
||||
= note: the matched value is of type `Foo<Void>`
|
||||
help: ensure that all possible cases are being handled by adding a match arm with a wildcard pattern as shown
|
||||
|
|
||||
LL ~ match x {
|
||||
LL + _ => todo!(),
|
||||
LL ~ }
|
||||
|
|
||||
|
||||
error: aborting due to 1 previous error
|
||||
|
||||
For more information about this error, try `rustc --explain E0004`.
|
||||
|
|
@ -1,29 +1,45 @@
|
|||
//@ aux-build: staged-api.rs
|
||||
//@ revisions: current exhaustive
|
||||
|
||||
//! The field of `Pin` used to be public, which would cause `Pin<Void>` to be uninhabited. To remedy
|
||||
//! this, we temporarily made it so unstable fields are always considered inhabited. This has now
|
||||
//! been reverted, and this file ensures that we don't special-case unstable fields wrt
|
||||
//! inhabitedness anymore.
|
||||
#![feature(exhaustive_patterns)]
|
||||
#![feature(never_type)]
|
||||
#![feature(my_coro_state)] // Custom feature from `staged-api.rs`
|
||||
#![deny(unreachable_patterns)]
|
||||
|
||||
extern crate staged_api;
|
||||
|
||||
use staged_api::Foo;
|
||||
use staged_api::{Foo, MyCoroutineState};
|
||||
|
||||
enum Void {}
|
||||
|
||||
fn demo(x: Foo<Void>) {
|
||||
match x {}
|
||||
//~^ ERROR non-exhaustive patterns
|
||||
}
|
||||
|
||||
// Ensure that the pattern is not considered unreachable.
|
||||
// Ensure that the pattern is considered unreachable.
|
||||
fn demo2(x: Foo<Void>) {
|
||||
match x {
|
||||
Foo { .. } => {}
|
||||
Foo { .. } => {} //~ ERROR unreachable
|
||||
}
|
||||
}
|
||||
|
||||
// Same as above, but for wildcard.
|
||||
fn demo3(x: Foo<Void>) {
|
||||
match x { _ => {} }
|
||||
match x {
|
||||
_ => {} //~ ERROR unreachable
|
||||
}
|
||||
}
|
||||
|
||||
fn unstable_enum(x: MyCoroutineState<i32, !>) {
|
||||
match x {
|
||||
MyCoroutineState::Yielded(_) => {}
|
||||
}
|
||||
match x {
|
||||
MyCoroutineState::Yielded(_) => {}
|
||||
MyCoroutineState::Complete(_) => {} //~ ERROR unreachable
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {}
|
||||
|
|
|
|||
40
tests/ui/uninhabited/uninhabited-unstable-field.stderr
Normal file
40
tests/ui/uninhabited/uninhabited-unstable-field.stderr
Normal file
|
|
@ -0,0 +1,40 @@
|
|||
error: unreachable pattern
|
||||
--> $DIR/uninhabited-unstable-field.rs:24:9
|
||||
|
|
||||
LL | Foo { .. } => {}
|
||||
| ^^^^^^^^^^------
|
||||
| |
|
||||
| matches no values because `Foo<Void>` is uninhabited
|
||||
| help: remove the match arm
|
||||
|
|
||||
= note: to learn more about uninhabited types, see https://doc.rust-lang.org/nomicon/exotic-sizes.html#empty-types
|
||||
note: the lint level is defined here
|
||||
--> $DIR/uninhabited-unstable-field.rs:9:9
|
||||
|
|
||||
LL | #![deny(unreachable_patterns)]
|
||||
| ^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
error: unreachable pattern
|
||||
--> $DIR/uninhabited-unstable-field.rs:31:9
|
||||
|
|
||||
LL | _ => {}
|
||||
| ^------
|
||||
| |
|
||||
| matches no values because `Foo<Void>` is uninhabited
|
||||
| help: remove the match arm
|
||||
|
|
||||
= note: to learn more about uninhabited types, see https://doc.rust-lang.org/nomicon/exotic-sizes.html#empty-types
|
||||
|
||||
error: unreachable pattern
|
||||
--> $DIR/uninhabited-unstable-field.rs:41:9
|
||||
|
|
||||
LL | MyCoroutineState::Complete(_) => {}
|
||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^------
|
||||
| |
|
||||
| matches no values because `!` is uninhabited
|
||||
| help: remove the match arm
|
||||
|
|
||||
= note: to learn more about uninhabited types, see https://doc.rust-lang.org/nomicon/exotic-sizes.html#empty-types
|
||||
|
||||
error: aborting due to 3 previous errors
|
||||
|
||||
Loading…
Add table
Add a link
Reference in a new issue