Skip to content

Commit 0afd16b

Browse files
committed
Start using pattern types in libcore
1 parent 6c6b492 commit 0afd16b

File tree

12 files changed

+130
-24
lines changed

12 files changed

+130
-24
lines changed

compiler/rustc_codegen_llvm/src/debuginfo/metadata.rs

+12-1
Original file line numberDiff line numberDiff line change
@@ -456,7 +456,18 @@ pub(crate) fn type_di_node<'ll, 'tcx>(cx: &CodegenCx<'ll, 'tcx>, t: Ty<'tcx>) ->
456456
AdtKind::Enum => enums::build_enum_type_di_node(cx, unique_type_id),
457457
},
458458
ty::Tuple(_) => build_tuple_type_di_node(cx, unique_type_id),
459-
_ => bug!("debuginfo: unexpected type in type_di_node(): {:?}", t),
459+
ty::Pat(base, _) => return type_di_node(cx, base),
460+
// FIXME(unsafe_binders): impl debug info
461+
ty::UnsafeBinder(_) => unimplemented!(),
462+
ty::Alias(..)
463+
| ty::Param(_)
464+
| ty::Bound(..)
465+
| ty::Infer(_)
466+
| ty::Placeholder(_)
467+
| ty::CoroutineWitness(..)
468+
| ty::Error(_) => {
469+
bug!("debuginfo: unexpected type in type_di_node(): {:?}", t)
470+
}
460471
};
461472

462473
{

compiler/rustc_lint/src/foreign_modules.rs

+1-4
Original file line numberDiff line numberDiff line change
@@ -241,10 +241,7 @@ fn structurally_same_type_impl<'tcx>(
241241
if let ty::Adt(def, args) = *ty.kind() {
242242
let is_transparent = def.repr().transparent();
243243
let is_non_null = types::nonnull_optimization_guaranteed(tcx, def);
244-
debug!(
245-
"non_transparent_ty({:?}) -- type is transparent? {}, type is non-null? {}",
246-
ty, is_transparent, is_non_null
247-
);
244+
debug!(?ty, is_transparent, is_non_null);
248245
if is_transparent && !is_non_null {
249246
debug_assert_eq!(def.variants().len(), 1);
250247
let v = &def.variant(FIRST_VARIANT);

compiler/rustc_lint/src/types.rs

+10-8
Original file line numberDiff line numberDiff line change
@@ -891,9 +891,13 @@ fn get_nullable_type<'tcx>(
891891
};
892892
return get_nullable_type(tcx, typing_env, inner_field_ty);
893893
}
894-
ty::Int(ty) => Ty::new_int(tcx, ty),
895-
ty::Uint(ty) => Ty::new_uint(tcx, ty),
896-
ty::RawPtr(ty, mutbl) => Ty::new_ptr(tcx, ty, mutbl),
894+
ty::Pat(base, pat)
895+
if let ty::PatternKind::Range { start: Some(start), end: None, .. } = *pat
896+
&& start.try_to_bits(tcx, typing_env) == Some(1) =>
897+
{
898+
base
899+
}
900+
ty::Int(_) | ty::Uint(_) | ty::RawPtr(..) => ty,
897901
// As these types are always non-null, the nullable equivalent of
898902
// `Option<T>` of these types are their raw pointer counterparts.
899903
ty::Ref(_region, ty, mutbl) => Ty::new_ptr(tcx, ty, mutbl),
@@ -1240,11 +1244,9 @@ impl<'a, 'tcx> ImproperCTypesVisitor<'a, 'tcx> {
12401244
help: Some(fluent::lint_improper_ctypes_char_help),
12411245
},
12421246

1243-
ty::Pat(..) => FfiUnsafe {
1244-
ty,
1245-
reason: fluent::lint_improper_ctypes_pat_reason,
1246-
help: Some(fluent::lint_improper_ctypes_pat_help),
1247-
},
1247+
// It's just extra invariants on the type that you need to uphold,
1248+
// but only the base type is relevant for being representable in FFI.
1249+
ty::Pat(base, ..) => self.check_type_for_ffi(acc, base),
12481250

12491251
ty::Int(ty::IntTy::I128) | ty::Uint(ty::UintTy::U128) => {
12501252
FfiUnsafe { ty, reason: fluent::lint_improper_ctypes_128bit, help: None }

compiler/rustc_mir_build/src/builder/matches/test.rs

+6
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,12 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
177177
place = ref_str;
178178
ty = ref_str_ty;
179179
}
180+
&ty::Pat(base, _) => {
181+
assert_eq!(ty, value.ty());
182+
183+
ty = base;
184+
expect_ty = base;
185+
}
180186
_ => {}
181187
}
182188

library/core/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,8 @@
181181
#![feature(no_core)]
182182
#![feature(no_sanitize)]
183183
#![feature(optimize_attribute)]
184+
#![feature(pattern_type_macro)]
185+
#![feature(pattern_types)]
184186
#![feature(prelude_import)]
185187
#![feature(repr_simd)]
186188
#![feature(rustc_allow_const_fn_unstable)]

library/core/src/num/niche_types.rs

+15-4
Original file line numberDiff line numberDiff line change
@@ -8,19 +8,28 @@ use crate::cmp::Ordering;
88
use crate::fmt;
99
use crate::hash::{Hash, Hasher};
1010
use crate::marker::StructuralPartialEq;
11+
#[cfg(not(bootstrap))]
12+
use crate::pattern_type;
1113

1214
macro_rules! define_valid_range_type {
1315
($(
1416
$(#[$m:meta])*
1517
$vis:vis struct $name:ident($int:ident as $uint:ident in $low:literal..=$high:literal);
1618
)+) => {$(
17-
#[derive(Clone, Copy, Eq)]
19+
#[derive(Clone, Copy)]
1820
#[repr(transparent)]
19-
#[rustc_layout_scalar_valid_range_start($low)]
20-
#[rustc_layout_scalar_valid_range_end($high)]
21+
#[cfg_attr(bootstrap, rustc_layout_scalar_valid_range_start($low))]
22+
#[cfg_attr(bootstrap, rustc_layout_scalar_valid_range_end($high))]
2123
$(#[$m])*
24+
#[cfg(bootstrap)]
2225
$vis struct $name($int);
2326

27+
#[derive(Clone, Copy)]
28+
#[repr(transparent)]
29+
$(#[$m])*
30+
#[cfg(not(bootstrap))]
31+
$vis struct $name(pattern_type!($int is $low..=$high));
32+
2433
const _: () = {
2534
// With the `valid_range` attributes, it's always specified as unsigned
2635
assert!(<$uint>::MIN == 0);
@@ -41,7 +50,7 @@ macro_rules! define_valid_range_type {
4150
#[inline]
4251
pub const unsafe fn new_unchecked(val: $int) -> Self {
4352
// SAFETY: Caller promised that `val` is non-zero.
44-
unsafe { $name(val) }
53+
unsafe { $name(crate::mem::transmute(val)) }
4554
}
4655

4756
#[inline]
@@ -57,6 +66,8 @@ macro_rules! define_valid_range_type {
5766
// by <https://github.com/rust-lang/compiler-team/issues/807>.
5867
impl StructuralPartialEq for $name {}
5968

69+
impl Eq for $name {}
70+
6071
impl PartialEq for $name {
6172
#[inline]
6273
fn eq(&self, other: &Self) -> bool {

tests/ui/consts/const-eval/raw-bytes.64bit.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ error[E0080]: it is undefined behavior to use this value
6868
--> $DIR/raw-bytes.rs:59:1
6969
|
7070
LL | const NULL_U8: NonZero<u8> = unsafe { mem::transmute(0u8) };
71-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0: encountered 0, but expected something greater or equal to 1
71+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0.0: encountered 0, but expected something greater or equal to 1
7272
|
7373
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rustc repository if you believe it should not be considered undefined behavior.
7474
= note: the raw bytes of the constant (size: 1, align: 1) {
@@ -79,7 +79,7 @@ error[E0080]: it is undefined behavior to use this value
7979
--> $DIR/raw-bytes.rs:61:1
8080
|
8181
LL | const NULL_USIZE: NonZero<usize> = unsafe { mem::transmute(0usize) };
82-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0: encountered 0, but expected something greater or equal to 1
82+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0.0: encountered 0, but expected something greater or equal to 1
8383
|
8484
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rustc repository if you believe it should not be considered undefined behavior.
8585
= note: the raw bytes of the constant (size: 8, align: 8) {

tests/ui/consts/const-eval/ub-nonnull.stderr

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ error[E0080]: it is undefined behavior to use this value
1919
--> $DIR/ub-nonnull.rs:24:1
2020
|
2121
LL | const NULL_U8: NonZero<u8> = unsafe { mem::transmute(0u8) };
22-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0: encountered 0, but expected something greater or equal to 1
22+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0.0: encountered 0, but expected something greater or equal to 1
2323
|
2424
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rustc repository if you believe it should not be considered undefined behavior.
2525
= note: the raw bytes of the constant (size: $SIZE, align: $ALIGN) {
@@ -30,7 +30,7 @@ error[E0080]: it is undefined behavior to use this value
3030
--> $DIR/ub-nonnull.rs:26:1
3131
|
3232
LL | const NULL_USIZE: NonZero<usize> = unsafe { mem::transmute(0usize) };
33-
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0: encountered 0, but expected something greater or equal to 1
33+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ constructing invalid value at .0.0: encountered 0, but expected something greater or equal to 1
3434
|
3535
= note: The rules on what exactly is undefined behavior aren't clear, so this check might be overzealous. Please open an issue on the rustc repository if you believe it should not be considered undefined behavior.
3636
= note: the raw bytes of the constant (size: $SIZE, align: $ALIGN) {

tests/ui/lint/clashing-extern-fn.stderr

+37-1
Original file line numberDiff line numberDiff line change
@@ -210,6 +210,30 @@ LL | fn non_null_ptr() -> *const usize;
210210
= note: expected `unsafe extern "C" fn() -> NonNull<usize>`
211211
found `unsafe extern "C" fn() -> *const usize`
212212

213+
warning: `option_non_zero_usize` redeclared with a different signature
214+
--> $DIR/clashing-extern-fn.rs:420:13
215+
|
216+
LL | fn option_non_zero_usize() -> usize;
217+
| ------------------------------------ `option_non_zero_usize` previously declared here
218+
...
219+
LL | fn option_non_zero_usize() -> Option<core::num::NonZero<usize>>;
220+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ this signature doesn't match the previous declaration
221+
|
222+
= note: expected `unsafe extern "C" fn() -> usize`
223+
found `unsafe extern "C" fn() -> Option<NonZero<usize>>`
224+
225+
warning: `option_non_zero_isize` redeclared with a different signature
226+
--> $DIR/clashing-extern-fn.rs:421:13
227+
|
228+
LL | fn option_non_zero_isize() -> isize;
229+
| ------------------------------------ `option_non_zero_isize` previously declared here
230+
...
231+
LL | fn option_non_zero_isize() -> Option<core::num::NonZero<isize>>;
232+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ this signature doesn't match the previous declaration
233+
|
234+
= note: expected `unsafe extern "C" fn() -> isize`
235+
found `unsafe extern "C" fn() -> Option<NonZero<isize>>`
236+
213237
warning: `option_non_zero_usize_incorrect` redeclared with a different signature
214238
--> $DIR/clashing-extern-fn.rs:425:13
215239
|
@@ -234,6 +258,18 @@ LL | fn option_non_null_ptr_incorrect() -> *const isize;
234258
= note: expected `unsafe extern "C" fn() -> *const usize`
235259
found `unsafe extern "C" fn() -> *const isize`
236260

261+
warning: `hidden_niche_transparent` redeclared with a different signature
262+
--> $DIR/clashing-extern-fn.rs:481:13
263+
|
264+
LL | fn hidden_niche_transparent() -> usize;
265+
| --------------------------------------- `hidden_niche_transparent` previously declared here
266+
...
267+
LL | fn hidden_niche_transparent() -> Option<Transparent>;
268+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ this signature doesn't match the previous declaration
269+
|
270+
= note: expected `unsafe extern "C" fn() -> usize`
271+
found `unsafe extern "C" fn() -> Option<Transparent>`
272+
237273
warning: `hidden_niche_transparent_no_niche` redeclared with a different signature
238274
--> $DIR/clashing-extern-fn.rs:483:13
239275
|
@@ -258,5 +294,5 @@ LL | fn hidden_niche_unsafe_cell() -> Option<UnsafeCell<NonZero<usiz
258294
= note: expected `unsafe extern "C" fn() -> usize`
259295
found `unsafe extern "C" fn() -> Option<UnsafeCell<NonZero<usize>>>`
260296

261-
warning: 22 warnings emitted
297+
warning: 25 warnings emitted
262298

tests/ui/lint/invalid_value.stderr

-2
Original file line numberDiff line numberDiff line change
@@ -333,7 +333,6 @@ LL | let _val: (NonZero<u32>, i32) = mem::uninitialized();
333333
|
334334
= note: `std::num::NonZero<u32>` must be non-null
335335
= note: because `core::num::niche_types::NonZeroU32Inner` must be non-null
336-
= note: integers must be initialized
337336

338337
error: the type `*const dyn Send` does not permit zero-initialization
339338
--> $DIR/invalid_value.rs:97:37
@@ -430,7 +429,6 @@ note: because `std::num::NonZero<u32>` must be non-null (in this field of the on
430429
LL | Banana(NonZero<u32>),
431430
| ^^^^^^^^^^^^
432431
= note: because `core::num::niche_types::NonZeroU32Inner` must be non-null
433-
= note: integers must be initialized
434432

435433
error: the type `bool` does not permit being left uninitialized
436434
--> $DIR/invalid_value.rs:111:26
+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
#![feature(pattern_types, pattern_type_macro, structural_match)]
2+
3+
use std::marker::StructuralPartialEq;
4+
use std::pat::pattern_type;
5+
6+
struct Thing(pattern_type!(u32 is 1..));
7+
8+
impl StructuralPartialEq for Thing {}
9+
impl Eq for Thing {}
10+
impl PartialEq for Thing {
11+
fn eq(&self, other: &Thing) -> bool {
12+
todo!()
13+
}
14+
}
15+
16+
const TWO: Thing = Thing(unsafe { std::mem::transmute(2_u32) });
17+
18+
const _: () = match TWO {
19+
TWO => {}
20+
_ => unreachable!(),
21+
};
22+
23+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
error[E0405]: cannot find trait `StructuralPartialEq` in this scope
2+
--> $DIR/matching.rs:7:6
3+
|
4+
LL | impl StructuralPartialEq for Thing {}
5+
| ^^^^^^^^^^^^^^^^^^^ not found in this scope
6+
|
7+
help: consider importing this trait
8+
|
9+
LL + use std::marker::StructuralPartialEq;
10+
|
11+
12+
error[E0405]: cannot find trait `StructuralEq` in this scope
13+
--> $DIR/matching.rs:8:6
14+
|
15+
LL | impl StructuralEq for Thing {}
16+
| ^^^^^^^^^^^^ not found in this scope
17+
18+
error: aborting due to 2 previous errors
19+
20+
For more information about this error, try `rustc --explain E0405`.

0 commit comments

Comments
 (0)