-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow blocks in const expressions #14183
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -47,7 +47,6 @@ use std::rc::Rc; | |
// fixed-size vectors and strings: [] and ""/_ | ||
// vector and string slices: &[] and &"" | ||
// tuples: (,) | ||
// records: {...} | ||
// enums: foo(...) | ||
// floating point literals and operators | ||
// & and * pointers | ||
|
@@ -241,6 +240,13 @@ impl<'a> ConstEvalVisitor<'a> { | |
|
||
ast::ExprRepeat(..) => general_const, | ||
|
||
ast::ExprBlock(ref block) => { | ||
match block.expr { | ||
Some(ref e) => self.classify(&**e), | ||
None => integral_const | ||
} | ||
} | ||
|
||
_ => non_const | ||
}; | ||
self.ccache.insert(did, cn); | ||
|
@@ -479,6 +485,12 @@ pub fn eval_const_expr_partial<T: ty::ExprTyProvider>(tcx: &T, e: &Expr) | |
// If we have a vstore, just keep going; it has to be a string | ||
ExprVstore(e, _) => eval_const_expr_partial(tcx, e), | ||
ExprParen(e) => eval_const_expr_partial(tcx, e), | ||
ExprBlock(ref block) => { | ||
match block.expr { | ||
Some(ref expr) => eval_const_expr_partial(tcx, &**expr), | ||
None => Ok(const_int(0i64)) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This seems odd, but it may just be that I'm unfamiliar with trans. Is this how we translate to There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, I didn't understand it myself, but other code treated unit that way, so I just copied it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah ok, nevermind then! |
||
} | ||
} | ||
_ => Err("unsupported constant expr".to_strbuf()) | ||
} | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
// Copyright 2014 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. | ||
|
||
pub static BLOCK_FN_DEF: fn(uint) -> uint = { | ||
fn foo(a: uint) -> uint { | ||
a + 10 | ||
} | ||
foo | ||
}; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
// Copyright 2014 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. | ||
|
||
#![feature(macro_rules)] | ||
|
||
static A: uint = { 1; 2 }; | ||
//~^ ERROR: blocks in constants are limited to items and tail expressions | ||
|
||
static B: uint = { { } 2 }; | ||
//~^ ERROR: blocks in constants are limited to items and tail expressions | ||
|
||
macro_rules! foo { | ||
() => (()) //~ ERROR: blocks in constants are limited to items and tail expressions | ||
} | ||
static C: uint = { foo!() 2 }; | ||
|
||
static D: uint = { let x = 4; 2 }; | ||
//~^ ERROR: blocks in constants are limited to items and tail expressions | ||
|
||
pub fn main() { | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
// Copyright 2014 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. | ||
|
||
// aux-build:cci_const_block.rs | ||
|
||
extern crate cci_const_block; | ||
|
||
pub fn main() { | ||
assert_eq!(cci_const_block::BLOCK_FN_DEF(390), 400); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
// Copyright 2014 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. | ||
|
||
// General test that function items in static blocks | ||
// can be generated with a macro. | ||
|
||
#![feature(macro_rules)] | ||
|
||
struct MyType { | ||
desc: &'static str, | ||
data: uint, | ||
code: fn(uint, uint) -> uint | ||
} | ||
|
||
impl MyType { | ||
fn eval(&self, a: uint) -> uint { | ||
(self.code)(self.data, a) | ||
} | ||
} | ||
|
||
macro_rules! codegen { | ||
($e:expr, $v:expr) => { | ||
{ | ||
fn generated(a: uint, b: uint) -> uint { | ||
a - ($e * b) | ||
} | ||
MyType { | ||
desc: "test", | ||
data: $v, | ||
code: generated | ||
} | ||
} | ||
} | ||
} | ||
|
||
static GENERATED_CODE_1: MyType = codegen!(2, 100); | ||
static GENERATED_CODE_2: MyType = codegen!(5, 1000); | ||
|
||
pub fn main() { | ||
assert_eq!(GENERATED_CODE_1.eval(10), 80); | ||
assert_eq!(GENERATED_CODE_2.eval(100), 500); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,56 @@ | ||
// Copyright 2014 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. | ||
|
||
#![feature(macro_rules)] | ||
|
||
mod foo { | ||
pub trait Value { | ||
fn value(&self) -> uint; | ||
} | ||
} | ||
|
||
static BLOCK_USE: uint = { | ||
use foo::Value; | ||
100 | ||
}; | ||
|
||
static BLOCK_PUB_USE: uint = { | ||
pub use foo::Value; | ||
200 | ||
}; | ||
|
||
static BLOCK_STRUCT_DEF: uint = { | ||
struct Foo { | ||
a: uint | ||
} | ||
Foo{ a: 300 }.a | ||
}; | ||
|
||
static BLOCK_FN_DEF: fn(uint) -> uint = { | ||
fn foo(a: uint) -> uint { | ||
a + 10 | ||
} | ||
foo | ||
}; | ||
|
||
static BLOCK_MACRO_RULES: uint = { | ||
macro_rules! baz { | ||
() => (412) | ||
} | ||
baz!() | ||
}; | ||
|
||
pub fn main() { | ||
assert_eq!(BLOCK_USE, 100); | ||
assert_eq!(BLOCK_PUB_USE, 200); | ||
assert_eq!(BLOCK_STRUCT_DEF, 300); | ||
assert_eq!(BLOCK_FN_DEF(390), 400); | ||
assert_eq!(BLOCK_MACRO_RULES, 412); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,69 @@ | ||
// Copyright 2014 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. | ||
|
||
#![allow(dead_code)] | ||
#![allow(unused_unsafe)] | ||
|
||
struct Foo { | ||
a: uint, | ||
b: *() | ||
} | ||
|
||
fn foo<T>(a: T) -> T { | ||
a | ||
} | ||
|
||
static BLOCK_INTEGRAL: uint = { 1 }; | ||
static BLOCK_EXPLICIT_UNIT: () = { () }; | ||
static BLOCK_IMPLICIT_UNIT: () = { }; | ||
static BLOCK_FLOAT: f64 = { 1.0 }; | ||
static BLOCK_ENUM: Option<uint> = { Some(100) }; | ||
static BLOCK_STRUCT: Foo = { Foo { a: 12, b: 0 as *() } }; | ||
static BLOCK_UNSAFE: uint = unsafe { 1000 }; | ||
|
||
// FIXME: #13970 | ||
// static BLOCK_FN_INFERRED: fn(uint) -> uint = { foo }; | ||
|
||
// FIXME: #13971 | ||
// static BLOCK_FN: fn(uint) -> uint = { foo::<uint> }; | ||
|
||
// FIXME: #13972 | ||
// static BLOCK_ENUM_CONSTRUCTOR: fn(uint) -> Option<uint> = { Some }; | ||
|
||
// FIXME: #13973 | ||
// static BLOCK_UNSAFE_SAFE_PTR: &'static int = unsafe { &*(0xdeadbeef as *int) }; | ||
// static BLOCK_UNSAFE_SAFE_PTR_2: &'static int = unsafe { | ||
// static X: *int = 0xdeadbeef as *int; | ||
// &*X | ||
// }; | ||
|
||
pub fn main() { | ||
assert_eq!(BLOCK_INTEGRAL, 1); | ||
assert_eq!(BLOCK_EXPLICIT_UNIT, ()); | ||
assert_eq!(BLOCK_IMPLICIT_UNIT, ()); | ||
assert_eq!(BLOCK_FLOAT, 1.0_f64); | ||
assert_eq!(BLOCK_STRUCT.a, 12); | ||
assert_eq!(BLOCK_STRUCT.b, 0 as *()); | ||
assert_eq!(BLOCK_ENUM, Some(100)); | ||
assert_eq!(BLOCK_UNSAFE, 1000); | ||
|
||
// FIXME: #13970 | ||
// assert_eq!(BLOCK_FN_INFERRED(300), 300); | ||
|
||
// FIXME: #13971 | ||
// assert_eq!(BLOCK_FN(300), 300); | ||
|
||
// FIXME: #13972 | ||
// assert_eq!(BLOCK_ENUM_CONSTRUCTOR(200), Some(200)); | ||
|
||
// FIXME: #13973 | ||
// assert_eq!(BLOCK_UNSAFE_SAFE_PTR as *int as uint, 0xdeadbeef_u); | ||
// assert_eq!(BLOCK_UNSAFE_SAFE_PTR_2 as *int as uint, 0xdeadbeef_u); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think
macro_rules!
macros may stick around un-expanded. Can you add a test case too make sure we don't ICE here?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you mean
macro_rules!
itself, or macros generated by it? For the latter there is a testcase already.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah,
macro_rules!
itselfThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added a testcase, doesn't seem to ICE. :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Excellent, thanks!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Or actually, can you add another test with the macro tagged with
#[macro_export]
? Those seem to be the ones that stick around.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
gives me the error
Am I doing something wrong?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh cool, looks like the parser doesn't even accept that currently (it wouldn't make much sense anyway I think). Thanks for checking!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I get the same error in a regular function as well: