Skip to content

Commit

Permalink
Auto merge of rust-lang#41055 - Archytaus:compile-fail/const-match-pa…
Browse files Browse the repository at this point in the history
…ttern-arm, r=arielb1

Fixed ICEs with pattern matching in const expression

Fixed 2 ICEs with when pattern matching inside a constant expression.

Both of these ICEs now resolve to an appropriate compiler error.

1. ICE was caused by a compiler bug to implement discriminant const qualify.

    I removed this intentionally thrown bug and changed it to a FIXME as the unimplemented expression type is handled as a compiler error elsewhere.

2. ICE was caused during a drop check when checking if a variable lifetime outlives the current scope if there was no parent scope .

    I've changed it to stop checking if there is no parent scope for the current scope. It is valid syntax for a const variable to be assigned a match expression with no enclosing scope.

    The ICE seemed to mainly be used as a defensive check for bugs elsewhere.

Fixes rust-lang#38199.
Fixes rust-lang#31577.
Fixes rust-lang#29093.
Fixes rust-lang#40012.
  • Loading branch information
bors committed Apr 8, 2017
2 parents fe39e94 + c9932b3 commit a610117
Show file tree
Hide file tree
Showing 3 changed files with 33 additions and 12 deletions.
11 changes: 2 additions & 9 deletions src/librustc_mir/transform/qualify_consts.rs
Original file line number Diff line number Diff line change
Expand Up @@ -603,7 +603,8 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
Rvalue::Cast(CastKind::ReifyFnPointer, ..) |
Rvalue::Cast(CastKind::UnsafeFnPointer, ..) |
Rvalue::Cast(CastKind::ClosureFnPointer, ..) |
Rvalue::Cast(CastKind::Unsize, ..) => {}
Rvalue::Cast(CastKind::Unsize, ..) |
Rvalue::Discriminant(..) => {}

Rvalue::Len(_) => {
// Static lvalues in consts would have errored already,
Expand Down Expand Up @@ -721,14 +722,6 @@ impl<'a, 'tcx> Visitor<'tcx> for Qualifier<'a, 'tcx, 'tcx> {
}
}

Rvalue::Discriminant(..) => {
// FIXME discriminant
self.add(Qualif::NOT_CONST);
if self.mode != Mode::Fn {
bug!("implement discriminant const qualify");
}
}

Rvalue::Box(_) => {
self.add(Qualif::NOT_CONST);
if self.mode != Mode::Fn {
Expand Down
9 changes: 6 additions & 3 deletions src/librustc_typeck/check/dropck.rs
Original file line number Diff line number Diff line change
Expand Up @@ -278,9 +278,12 @@ pub fn check_safety_of_destructor_if_necessary<'a, 'gcx, 'tcx>(
debug!("check_safety_of_destructor_if_necessary typ: {:?} scope: {:?}",
typ, scope);

let parent_scope = rcx.tcx.region_maps.opt_encl_scope(scope).unwrap_or_else(|| {
span_bug!(span, "no enclosing scope found for scope: {:?}", scope)
});

let parent_scope = match rcx.tcx.region_maps.opt_encl_scope(scope) {
Some(parent_scope) => parent_scope,
// If no enclosing scope, then it must be the root scope which cannot be outlived.
None => return
};

let result = iterate_over_potentially_unsafe_regions_in_type(
&mut DropckContext {
Expand Down
25 changes: 25 additions & 0 deletions src/test/compile-fail/const-match-pattern-arm.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

const x: bool = match Some(true) {
Some(value) => true,
//~^ ERROR: constant contains unimplemented expression type [E0019]
_ => false
};

const y: bool = {
match Some(true) {
Some(value) => true,
//~^ ERROR: constant contains unimplemented expression type [E0019]
_ => false
}
};

fn main() {}

0 comments on commit a610117

Please sign in to comment.