Skip to content

Commit 7c301ec

Browse files
authored
Rollup merge of rust-lang#134371 - scottmcm:fix-134352, r=oli-obk
Check for array lengths that aren't actually `usize` I wish typeck wouldn't give us `ty::Array`s that have this problem in the first place, but we can check for it. Fixes rust-lang#134352 cc ``@matthiaskrgr``
2 parents dffaad8 + 43a79a0 commit 7c301ec

File tree

3 files changed

+48
-7
lines changed

3 files changed

+48
-7
lines changed

compiler/rustc_mir_build/src/build/expr/as_place.rs

+25-7
Original file line numberDiff line numberDiff line change
@@ -647,13 +647,31 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
647647

648648
match place_ty.kind() {
649649
ty::Array(_elem_ty, len_const) => {
650-
// We know how long an array is, so just use that as a constant
651-
// directly -- no locals needed. We do need one statement so
652-
// that borrow- and initialization-checking consider it used,
653-
// though. FIXME: Do we really *need* to count this as a use?
654-
// Could partial array tracking work off something else instead?
655-
self.cfg.push_fake_read(block, source_info, FakeReadCause::ForIndex, place);
656-
let const_ = Const::from_ty_const(*len_const, usize_ty, self.tcx);
650+
let ty_const = if let Some((_, len_ty)) = len_const.try_to_valtree()
651+
&& len_ty != self.tcx.types.usize
652+
{
653+
// Bad const generics can give us a constant from the type that's
654+
// not actually a `usize`, so in that case give an error instead.
655+
// FIXME: It'd be nice if the type checker made sure this wasn't
656+
// possible, instead.
657+
let err = self.tcx.dcx().span_delayed_bug(
658+
span,
659+
format!(
660+
"Array length should have already been a type error, as it's {len_ty:?}"
661+
),
662+
);
663+
ty::Const::new_error(self.tcx, err)
664+
} else {
665+
// We know how long an array is, so just use that as a constant
666+
// directly -- no locals needed. We do need one statement so
667+
// that borrow- and initialization-checking consider it used,
668+
// though. FIXME: Do we really *need* to count this as a use?
669+
// Could partial array tracking work off something else instead?
670+
self.cfg.push_fake_read(block, source_info, FakeReadCause::ForIndex, place);
671+
*len_const
672+
};
673+
674+
let const_ = Const::from_ty_const(ty_const, usize_ty, self.tcx);
657675
Operand::Constant(Box::new(ConstOperand { span, user_ty: None, const_ }))
658676
}
659677
ty::Slice(_elem_ty) => {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
//@ check-fail
2+
//@ compile-flags: -C opt-level=0
3+
4+
#![crate_type = "lib"]
5+
6+
// This used to fail in the known-panics lint, as the MIR was ill-typed due to
7+
// the length constant not actually having type usize.
8+
// https://github.com/rust-lang/rust/issues/134352
9+
10+
pub struct BadStruct<const N: i64>(pub [u8; N]);
11+
//~^ ERROR: the constant `N` is not of type `usize`
12+
13+
pub fn bad_array_length_type(value: BadStruct<3>) -> u8 {
14+
value.0[0]
15+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
error: the constant `N` is not of type `usize`
2+
--> $DIR/index_array_bad_type.rs:10:40
3+
|
4+
LL | pub struct BadStruct<const N: i64>(pub [u8; N]);
5+
| ^^^^^^^ expected `usize`, found `i64`
6+
7+
error: aborting due to 1 previous error
8+

0 commit comments

Comments
 (0)