Skip to content

Commit eb1d018

Browse files
committed
Auto merge of #25570 - oli-obk:const_indexing, r=nikomatsakis
This PR allows the constant evaluation of index operations on constant arrays and repeat expressions. This allows index expressions to appear in the expression path of the length expression of a repeat expression or an array type. An example is ```rust const ARR: [usize; 5] = [1, 2, 3, 4, 5]; const ARR2: [usize; ARR[1]] = [42, 99]; ``` In most other locations llvm's const evaluator figures it out already. This is not specific to index expressions and could be remedied in the future.
2 parents dfe88bf + 6405122 commit eb1d018

11 files changed

+212
-28
lines changed

src/librustc/middle/const_eval.rs

+93-1
Original file line numberDiff line numberDiff line change
@@ -253,6 +253,8 @@ pub enum ConstVal {
253253
Struct(ast::NodeId),
254254
Tuple(ast::NodeId),
255255
Function(DefId),
256+
Array(ast::NodeId, u64),
257+
Repeat(ast::NodeId, u64),
256258
}
257259

258260
impl hash::Hash for ConstVal {
@@ -267,6 +269,8 @@ impl hash::Hash for ConstVal {
267269
Struct(a) => a.hash(state),
268270
Tuple(a) => a.hash(state),
269271
Function(a) => a.hash(state),
272+
Array(a, n) => { a.hash(state); n.hash(state) },
273+
Repeat(a, n) => { a.hash(state); n.hash(state) },
270274
}
271275
}
272276
}
@@ -287,6 +291,8 @@ impl PartialEq for ConstVal {
287291
(&Struct(a), &Struct(b)) => a == b,
288292
(&Tuple(a), &Tuple(b)) => a == b,
289293
(&Function(a), &Function(b)) => a == b,
294+
(&Array(a, an), &Array(b, bn)) => (a == b) && (an == bn),
295+
(&Repeat(a, an), &Repeat(b, bn)) => (a == b) && (an == bn),
290296
_ => false,
291297
}
292298
}
@@ -307,6 +313,8 @@ impl ConstVal {
307313
Struct(_) => "struct",
308314
Tuple(_) => "tuple",
309315
Function(_) => "function definition",
316+
Array(..) => "array",
317+
Repeat(..) => "repeat",
310318
}
311319
}
312320
}
@@ -415,6 +423,12 @@ pub enum ErrKind {
415423
ExpectedConstTuple,
416424
ExpectedConstStruct,
417425
TupleIndexOutOfBounds,
426+
IndexedNonVec,
427+
IndexNegative,
428+
IndexNotInt,
429+
IndexOutOfBounds,
430+
RepeatCountNotNatural,
431+
RepeatCountNotInt,
418432

419433
MiscBinaryOp,
420434
MiscCatchAll,
@@ -455,6 +469,12 @@ impl ConstEvalErr {
455469
ExpectedConstTuple => "expected constant tuple".into_cow(),
456470
ExpectedConstStruct => "expected constant struct".into_cow(),
457471
TupleIndexOutOfBounds => "tuple index out of bounds".into_cow(),
472+
IndexedNonVec => "indexing is only supported for arrays".into_cow(),
473+
IndexNegative => "indices must be non-negative integers".into_cow(),
474+
IndexNotInt => "indices must be integers".into_cow(),
475+
IndexOutOfBounds => "array index out of bounds".into_cow(),
476+
RepeatCountNotNatural => "repeat count must be a natural number".into_cow(),
477+
RepeatCountNotInt => "repeat count must be integers".into_cow(),
458478

459479
MiscBinaryOp => "bad operands for binary".into_cow(),
460480
MiscCatchAll => "unsupported constant expr".into_cow(),
@@ -1111,11 +1131,83 @@ pub fn eval_const_expr_partial<'tcx>(tcx: &ty::ctxt<'tcx>,
11111131
hir::ExprBlock(ref block) => {
11121132
match block.expr {
11131133
Some(ref expr) => try!(eval_const_expr_partial(tcx, &**expr, ty_hint, fn_args)),
1114-
None => Int(0)
1134+
None => unreachable!(),
11151135
}
11161136
}
11171137
hir::ExprTup(_) => Tuple(e.id),
11181138
hir::ExprStruct(..) => Struct(e.id),
1139+
hir::ExprIndex(ref arr, ref idx) => {
1140+
if !tcx.sess.features.borrow().const_indexing {
1141+
tcx.sess.span_err(
1142+
e.span,
1143+
"const indexing is an unstable feature");
1144+
fileline_help!(
1145+
tcx.sess,
1146+
e.span,
1147+
"in Nightly builds, add `#![feature(const_indexing)]` to the crate \
1148+
attributes to enable");
1149+
signal!(e, NonConstPath)
1150+
}
1151+
let arr_hint = if let ExprTypeChecked = ty_hint {
1152+
ExprTypeChecked
1153+
} else {
1154+
UncheckedExprNoHint
1155+
};
1156+
let arr = try!(eval_const_expr_partial(tcx, arr, arr_hint, fn_args));
1157+
let idx_hint = if let ExprTypeChecked = ty_hint {
1158+
ExprTypeChecked
1159+
} else {
1160+
UncheckedExprHint(tcx.types.usize)
1161+
};
1162+
let idx = match try!(eval_const_expr_partial(tcx, idx, idx_hint, fn_args)) {
1163+
Int(i) if i >= 0 => i as u64,
1164+
Int(_) => signal!(idx, IndexNegative),
1165+
Uint(i) => i,
1166+
_ => signal!(idx, IndexNotInt),
1167+
};
1168+
match arr {
1169+
Array(_, n) if idx >= n => signal!(e, IndexOutOfBounds),
1170+
Array(v, _) => if let hir::ExprVec(ref v) = tcx.map.expect_expr(v).node {
1171+
try!(eval_const_expr_partial(tcx, &*v[idx as usize], ty_hint, fn_args))
1172+
} else {
1173+
unreachable!()
1174+
},
1175+
1176+
Repeat(_, n) if idx >= n => signal!(e, IndexOutOfBounds),
1177+
Repeat(elem, _) => try!(eval_const_expr_partial(
1178+
tcx,
1179+
&*tcx.map.expect_expr(elem),
1180+
ty_hint,
1181+
fn_args,
1182+
)),
1183+
1184+
ByteStr(ref data) if idx as usize >= data.len()
1185+
=> signal!(e, IndexOutOfBounds),
1186+
ByteStr(data) => Uint(data[idx as usize] as u64),
1187+
1188+
Str(ref s) if idx as usize >= s.len()
1189+
=> signal!(e, IndexOutOfBounds),
1190+
Str(_) => unimplemented!(), // there's no const_char type
1191+
_ => signal!(e, IndexedNonVec),
1192+
}
1193+
}
1194+
hir::ExprVec(ref v) => Array(e.id, v.len() as u64),
1195+
hir::ExprRepeat(_, ref n) => {
1196+
let len_hint = if let ExprTypeChecked = ty_hint {
1197+
ExprTypeChecked
1198+
} else {
1199+
UncheckedExprHint(tcx.types.usize)
1200+
};
1201+
Repeat(
1202+
e.id,
1203+
match try!(eval_const_expr_partial(tcx, &**n, len_hint, fn_args)) {
1204+
Int(i) if i >= 0 => i as u64,
1205+
Int(_) => signal!(e, RepeatCountNotNatural),
1206+
Uint(i) => i,
1207+
_ => signal!(e, RepeatCountNotInt),
1208+
},
1209+
)
1210+
},
11191211
hir::ExprTupField(ref base, index) => {
11201212
let base_hint = if let ExprTypeChecked = ty_hint {
11211213
ExprTypeChecked

src/librustc_trans/trans/mir/constant.rs

+6
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,12 @@ impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
6262
ConstVal::Function(_) => {
6363
unimplemented!()
6464
}
65+
ConstVal::Array(..) => {
66+
unimplemented!()
67+
}
68+
ConstVal::Repeat(..) => {
69+
unimplemented!()
70+
}
6571
};
6672

6773
assert!(!ty.has_erasable_regions());

src/libsyntax/feature_gate.rs

+6
Original file line numberDiff line numberDiff line change
@@ -179,6 +179,9 @@ const KNOWN_FEATURES: &'static [(&'static str, &'static str, Option<u32>, Status
179179
// Allows the definition of `const fn` functions.
180180
("const_fn", "1.2.0", Some(24111), Active),
181181

182+
// Allows indexing into constant arrays.
183+
("const_indexing", "1.4.0", Some(29947), Active),
184+
182185
// Allows using #[prelude_import] on glob `use` items.
183186
//
184187
// rustc internal
@@ -492,6 +495,7 @@ pub struct Features {
492495
/// #![feature] attrs for non-language (library) features
493496
pub declared_lib_features: Vec<(InternedString, Span)>,
494497
pub const_fn: bool,
498+
pub const_indexing: bool,
495499
pub static_recursion: bool,
496500
pub default_type_parameter_fallback: bool,
497501
pub type_macros: bool,
@@ -524,6 +528,7 @@ impl Features {
524528
declared_stable_lang_features: Vec::new(),
525529
declared_lib_features: Vec::new(),
526530
const_fn: false,
531+
const_indexing: false,
527532
static_recursion: false,
528533
default_type_parameter_fallback: false,
529534
type_macros: false,
@@ -1097,6 +1102,7 @@ fn check_crate_inner<F>(cm: &CodeMap, span_handler: &SpanHandler,
10971102
declared_stable_lang_features: accepted_features,
10981103
declared_lib_features: unknown_features,
10991104
const_fn: cx.has_feature("const_fn"),
1105+
const_indexing: cx.has_feature("const_indexing"),
11001106
static_recursion: cx.has_feature("static_recursion"),
11011107
default_type_parameter_fallback: cx.has_feature("default_type_parameter_fallback"),
11021108
type_macros: cx.has_feature("type_macros"),
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
static A: &'static [i32] = &[];
12+
static B: i32 = (&A)[1]; //~ ERROR: const index-expr is out of bounds
13+
14+
fn main() {
15+
let _ = B;
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
const A: [i32; 0] = [];
12+
const B: i32 = A[1]; //~ ERROR: const index-expr is out of bounds
13+
14+
fn main() {
15+
let _ = B;
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(const_indexing)]
12+
13+
const ARR: [i32; 6] = [42, 43, 44, 45, 46, 47];
14+
const IDX: usize = 3;
15+
const VAL: i32 = ARR[IDX];
16+
const BONG: [i32; (ARR[0] - 41) as usize] = [5];
17+
const BLUB: [i32; (ARR[0] - 40) as usize] = [5]; //~ ERROR: mismatched types
18+
const BOO: [i32; (ARR[0] - 41) as usize] = [5, 99]; //~ ERROR: mismatched types
19+
20+
fn main() {
21+
let _ = VAL;
22+
}

src/test/compile-fail/const-array-oob.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,13 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![feature(const_indexing)]
12+
1113
const FOO: [u32; 3] = [1, 2, 3];
12-
const BAR: u32 = FOO[5]; //~ ERROR const index-expr is out of bounds
14+
const BAR: u32 = FOO[5]; // no error, because the error below occurs before regular const eval
15+
16+
const BLUB: [u32; FOO[4]] = [5, 6];
17+
//~^ ERROR array length constant evaluation error: array index out of bounds [E0250]
1318

1419
fn main() {
1520
let _ = BAR;
+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(const_indexing)]
12+
13+
fn main() {
14+
const ARR: [i32; 6] = [42, 43, 44, 45, 46, 47];
15+
const IDX: usize = 3;
16+
const VAL: i32 = ARR[IDX];
17+
const BLUB: [i32; (ARR[0] - 41) as usize] = [5];
18+
}

src/test/run-pass/check-static-slice.rs

+24-21
Original file line numberDiff line numberDiff line change
@@ -12,30 +12,33 @@
1212
// and unsized) work properly.
1313

1414

15-
const aa: [isize; 3] = [1, 2, 3];
16-
const ab: &'static [isize; 3] = &aa;
17-
const ac: &'static [isize] = ab;
18-
const ad: &'static [isize] = &aa;
19-
const ae: &'static [isize; 3] = &[1, 2, 3];
20-
const af: &'static [isize] = &[1, 2, 3];
15+
const AA: [isize; 3] = [1, 2, 3];
16+
const AB: &'static [isize; 3] = &AA;
17+
const AC: &'static [isize] = AB;
18+
const AD: &'static [isize] = &AA;
19+
const AE: &'static [isize; 3] = &[1, 2, 3];
20+
const AF: &'static [isize] = &[1, 2, 3];
2121

22-
static ca: isize = aa[0];
23-
static cb: isize = ab[1];
24-
static cc: isize = ac[2];
25-
static cd: isize = ad[0];
26-
static ce: isize = ae[1];
27-
static cf: isize = af[2];
22+
static CA: isize = AA[0];
23+
static CB: isize = AB[1];
24+
static CC: isize = AC[2];
25+
static CD: isize = AD[0];
26+
static CE: isize = AE[1];
27+
static CF: isize = AF[2];
28+
29+
static AG: &'static isize = &AA[2];
2830

2931
fn main () {
3032
let b: &[isize] = &[1, 2, 3];
31-
assert_eq!(ac, b);
32-
assert_eq!(ad, b);
33-
assert_eq!(af, b);
33+
assert_eq!(AC, b);
34+
assert_eq!(AD, b);
35+
assert_eq!(AF, b);
36+
assert_eq!(*AG, 3);
3437

35-
assert_eq!(ca, 1);
36-
assert_eq!(cb, 2);
37-
assert_eq!(cc, 3);
38-
assert_eq!(cd, 1);
39-
assert_eq!(ce, 2);
40-
assert_eq!(cf, 3);
38+
assert_eq!(CA, 1);
39+
assert_eq!(CB, 2);
40+
assert_eq!(CC, 3);
41+
assert_eq!(CD, 1);
42+
assert_eq!(CE, 2);
43+
assert_eq!(CF, 3);
4144
}

src/test/run-pass/const-enum-vec-index.rs

+4-5
Original file line numberDiff line numberDiff line change
@@ -8,14 +8,13 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11-
1211
enum E { V1(isize), V0 }
1312
const C: &'static [E] = &[E::V0, E::V1(0xDEADBEE)];
1413
static C0: E = C[0];
1514
static C1: E = C[1];
16-
const D: &'static [E; 2] = &[E::V0, E::V1(0xDEADBEE)];
17-
static D0: E = C[0];
18-
static D1: E = C[1];
15+
const D: &'static [E; 2] = &[E::V0, E::V1(0xDEAFBEE)];
16+
static D0: E = D[0];
17+
static D1: E = D[1];
1918

2019
pub fn main() {
2120
match C0 {
@@ -32,7 +31,7 @@ pub fn main() {
3231
_ => panic!()
3332
}
3433
match D1 {
35-
E::V1(n) => assert_eq!(n, 0xDEADBEE),
34+
E::V1(n) => assert_eq!(n, 0xDEAFBEE),
3635
_ => panic!()
3736
}
3837
}

src/test/run-pass/const-str-ptr.rs

+1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ const C: *const u8 = B as *const u8;
1818
pub fn main() {
1919
unsafe {
2020
let foo = &A as *const u8;
21+
assert_eq!(foo, C);
2122
assert_eq!(str::from_utf8_unchecked(&A), "hi");
2223
assert_eq!(*C, A[0]);
2324
assert_eq!(*(&B[0] as *const u8), A[0]);

0 commit comments

Comments
 (0)