Fix an ICE using break and continue as array lengths
This commit is contained in:
parent
c6bbee802a
commit
7ad1c62d38
7 changed files with 60 additions and 6 deletions
|
@ -1211,6 +1211,7 @@ where
|
||||||
.set(derive_registrar::find(&hir_map));
|
.set(derive_registrar::find(&hir_map));
|
||||||
|
|
||||||
time(sess, "loop checking", || loops::check_crate(sess, &hir_map));
|
time(sess, "loop checking", || loops::check_crate(sess, &hir_map));
|
||||||
|
sess.abort_if_errors();
|
||||||
|
|
||||||
let mut local_providers = ty::query::Providers::default();
|
let mut local_providers = ty::query::Providers::default();
|
||||||
default_provide(&mut local_providers);
|
default_provide(&mut local_providers);
|
||||||
|
|
|
@ -541,9 +541,9 @@ impl<'a, 'gcx, 'tcx> Builder<'a, 'gcx, 'tcx> {
|
||||||
/// Finds the breakable scope for a given label. This is used for
|
/// Finds the breakable scope for a given label. This is used for
|
||||||
/// resolving `break` and `continue`.
|
/// resolving `break` and `continue`.
|
||||||
pub fn find_breakable_scope(&self,
|
pub fn find_breakable_scope(&self,
|
||||||
span: Span,
|
span: Span,
|
||||||
label: region::Scope)
|
label: region::Scope)
|
||||||
-> &BreakableScope<'tcx> {
|
-> &BreakableScope<'tcx> {
|
||||||
// find the loop-scope with the correct id
|
// find the loop-scope with the correct id
|
||||||
self.breakable_scopes.iter()
|
self.breakable_scopes.iter()
|
||||||
.rev()
|
.rev()
|
||||||
|
|
|
@ -40,6 +40,7 @@ enum Context {
|
||||||
Loop(LoopKind),
|
Loop(LoopKind),
|
||||||
Closure,
|
Closure,
|
||||||
LabeledBlock,
|
LabeledBlock,
|
||||||
|
AnonConst,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
|
@ -71,6 +72,10 @@ impl<'a, 'hir> Visitor<'hir> for CheckLoopVisitor<'a, 'hir> {
|
||||||
self.with_context(Normal, |v| intravisit::walk_impl_item(v, i));
|
self.with_context(Normal, |v| intravisit::walk_impl_item(v, i));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn visit_anon_const(&mut self, c: &'hir hir::AnonConst) {
|
||||||
|
self.with_context(AnonConst, |v| intravisit::walk_anon_const(v, c));
|
||||||
|
}
|
||||||
|
|
||||||
fn visit_expr(&mut self, e: &'hir hir::Expr) {
|
fn visit_expr(&mut self, e: &'hir hir::Expr) {
|
||||||
match e.node {
|
match e.node {
|
||||||
hir::ExprWhile(ref e, ref b, _) => {
|
hir::ExprWhile(ref e, ref b, _) => {
|
||||||
|
@ -194,7 +199,7 @@ impl<'a, 'hir> CheckLoopVisitor<'a, 'hir> {
|
||||||
.span_label(span, "cannot break inside of a closure")
|
.span_label(span, "cannot break inside of a closure")
|
||||||
.emit();
|
.emit();
|
||||||
}
|
}
|
||||||
Normal => {
|
Normal | AnonConst => {
|
||||||
struct_span_err!(self.sess, span, E0268, "`{}` outside of loop", name)
|
struct_span_err!(self.sess, span, E0268, "`{}` outside of loop", name)
|
||||||
.span_label(span, "cannot break outside of a loop")
|
.span_label(span, "cannot break outside of a loop")
|
||||||
.emit();
|
.emit();
|
||||||
|
|
19
src/test/ui/array-break-length.rs
Normal file
19
src/test/ui/array-break-length.rs
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
// Copyright 2018 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.
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
loop {
|
||||||
|
|_: [_; break]| {} //~ ERROR: `break` outside of loop
|
||||||
|
}
|
||||||
|
|
||||||
|
loop {
|
||||||
|
|_: [_; continue]| {} //~ ERROR: `continue` outside of loop
|
||||||
|
}
|
||||||
|
}
|
15
src/test/ui/array-break-length.stderr
Normal file
15
src/test/ui/array-break-length.stderr
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
error[E0268]: `break` outside of loop
|
||||||
|
--> $DIR/array-break-length.rs:13:17
|
||||||
|
|
|
||||||
|
LL | |_: [_; break]| {} //~ ERROR: `break` outside of loop
|
||||||
|
| ^^^^^ cannot break outside of a loop
|
||||||
|
|
||||||
|
error[E0268]: `continue` outside of loop
|
||||||
|
--> $DIR/array-break-length.rs:17:17
|
||||||
|
|
|
||||||
|
LL | |_: [_; continue]| {} //~ ERROR: `continue` outside of loop
|
||||||
|
| ^^^^^^^^ cannot break outside of a loop
|
||||||
|
|
||||||
|
error: aborting due to 2 previous errors
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0268`.
|
|
@ -12,6 +12,8 @@ fn main() {
|
||||||
|_: [_; continue]| {}; //~ ERROR: `continue` outside of loop
|
|_: [_; continue]| {}; //~ ERROR: `continue` outside of loop
|
||||||
|
|
||||||
while |_: [_; continue]| {} {} //~ ERROR: `break` or `continue` with no label
|
while |_: [_; continue]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
|
//~^ ERROR: `continue` outside of loop
|
||||||
|
|
||||||
while |_: [_; break]| {} {} //~ ERROR: `break` or `continue` with no label
|
while |_: [_; break]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
|
//~^ ERROR: `break` outside of loop
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,13 +10,25 @@ error[E0590]: `break` or `continue` with no label in the condition of a `while`
|
||||||
LL | while |_: [_; continue]| {} {} //~ ERROR: `break` or `continue` with no label
|
LL | while |_: [_; continue]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
| ^^^^^^^^ unlabeled `continue` in the condition of a `while` loop
|
| ^^^^^^^^ unlabeled `continue` in the condition of a `while` loop
|
||||||
|
|
||||||
|
error[E0268]: `continue` outside of loop
|
||||||
|
--> $DIR/closure-array-break-length.rs:14:19
|
||||||
|
|
|
||||||
|
LL | while |_: [_; continue]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
|
| ^^^^^^^^ cannot break outside of a loop
|
||||||
|
|
||||||
error[E0590]: `break` or `continue` with no label in the condition of a `while` loop
|
error[E0590]: `break` or `continue` with no label in the condition of a `while` loop
|
||||||
--> $DIR/closure-array-break-length.rs:16:19
|
--> $DIR/closure-array-break-length.rs:17:19
|
||||||
|
|
|
|
||||||
LL | while |_: [_; break]| {} {} //~ ERROR: `break` or `continue` with no label
|
LL | while |_: [_; break]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
| ^^^^^ unlabeled `break` in the condition of a `while` loop
|
| ^^^^^ unlabeled `break` in the condition of a `while` loop
|
||||||
|
|
||||||
error: aborting due to 3 previous errors
|
error[E0268]: `break` outside of loop
|
||||||
|
--> $DIR/closure-array-break-length.rs:17:19
|
||||||
|
|
|
||||||
|
LL | while |_: [_; break]| {} {} //~ ERROR: `break` or `continue` with no label
|
||||||
|
| ^^^^^ cannot break outside of a loop
|
||||||
|
|
||||||
|
error: aborting due to 5 previous errors
|
||||||
|
|
||||||
Some errors occurred: E0268, E0590.
|
Some errors occurred: E0268, E0590.
|
||||||
For more information about an error, try `rustc --explain E0268`.
|
For more information about an error, try `rustc --explain E0268`.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue