Closed

Description
I am writing some code which parses on-disk structures which vary their type based on magic numbers scattered through the structure. I have chosen to model this as a union of structs, and the magic number fields as byte arrays because they have neither power-of-two alignment or size, and then pattern-match on magic numbers to determine the type. This turns out to be sufficiently pathological to cause compiler panics.
Here is a minimal case that triggers the panic:
pub struct Punned {
foo: [u8; 1],
bar: [u8; 1],
}
pub fn test(punned: Punned) {
match punned {
Punned { foo: [_], .. } => println!("foo"),
Punned { bar: [_], .. } => println!("bar"),
}
}
Playing around with the different things being matched on — my real code matches on constants — suggest that this is a bug in the exhaustiveness checking.
Output of rustc --version --verbose
:
rustc 1.33.0-nightly (d22fa2d87 2019-01-08)
binary: rustc
commit-hash: d22fa2d87d03d19fdb1359faab9ec5e74eff26b3
commit-date: 2019-01-08
host: x86_64-apple-darwin
release: 1.33.0-nightly
LLVM version: 8.0
Output of RUST_BACKTRACE=1 rustc crash.rs
:
thread 'rustc' panicked at 'expected `LazyConst` to contain a usize', src/libcore/option.rs:1040:5
note: Some details are omitted, run with `RUST_BACKTRACE=full` for a verbose backtrace.
stack backtrace:
0: std::sys::unix::backtrace::tracing::imp::unwind_backtrace
1: std::sys_common::backtrace::_print
2: std::panicking::default_hook::{{closure}}
3: std::panicking::default_hook
4: rustc::util::common::panic_hook
5: std::panicking::rust_panic_with_hook
6: std::panicking::continue_panic_fmt
7: rust_begin_unwind
8: core::panicking::panic_fmt
9: core::option::expect_failed
10: rustc_mir::hair::pattern::_match::pat_constructors
11: rustc_mir::hair::pattern::_match::is_useful
12: rustc_mir::hair::pattern::_match::is_useful
13: rustc_mir::hair::pattern::_match::is_useful_specialized
14: <core::iter::Map<I, F> as core::iter::iterator::Iterator>::try_fold
15: rustc_mir::hair::pattern::_match::is_useful
16: rustc_mir::hair::pattern::_match::is_useful_specialized
17: <core::iter::Map<I, F> as core::iter::iterator::Iterator>::try_fold
18: rustc_mir::hair::pattern::_match::is_useful
19: rustc_mir::hair::pattern::check_match::check_arms
20: rustc_mir::hair::pattern::_match::MatchCheckCtxt::create_and_enter
21: <rustc_mir::hair::pattern::check_match::MatchVisitor<'a, 'tcx> as rustc::hir::intravisit::Visitor<'tcx>>::visit_expr
22: <rustc_mir::hair::pattern::check_match::MatchVisitor<'a, 'tcx> as rustc::hir::intravisit::Visitor<'tcx>>::visit_expr
23: <rustc_mir::hair::pattern::check_match::MatchVisitor<'a, 'tcx> as rustc::hir::intravisit::Visitor<'tcx>>::visit_body
24: rustc::session::Session::track_errors
25: rustc_mir::hair::pattern::check_match::check_match
26: rustc::ty::query::__query_compute::check_match
27: rustc::ty::query::<impl rustc::ty::query::config::QueryAccessors<'tcx> for rustc::ty::query::queries::check_match<'tcx>>::compute
28: rustc::dep_graph::graph::DepGraph::with_task_impl
29: rustc::ty::query::plumbing::<impl rustc::ty::context::TyCtxt<'a, 'gcx, 'tcx>>::try_get_with
30: <rustc_mir::hair::pattern::check_match::OuterVisitor<'a, 'tcx> as rustc::hir::intravisit::Visitor<'tcx>>::visit_body
31: rustc::hir::intravisit::walk_item
32: rustc::hir::Crate::visit_all_item_likes
33: rustc_mir::hair::pattern::check_match::check_crate
34: rustc::util::common::time
35: <std::thread::local::LocalKey<T>>::with
36: rustc_driver::driver::compile_input
37: rustc_driver::run_compiler_with_pool
38: <scoped_tls::ScopedKey<T>>::set
39: rustc_driver::run_compiler
40: <scoped_tls::ScopedKey<T>>::set
query stack during panic:
#0 [check_match] processing `test`
end of query stack
error: internal compiler error: unexpected panic
note: the compiler unexpectedly panicked. this is a bug.
note: we would appreciate a bug report: https://github.com/rust-lang/rust/blob/master/CONTRIBUTING.md#bug-reports
note: rustc 1.33.0-nightly (d22fa2d87 2019-01-08) running on x86_64-apple-darwin