Implement RFC 1440 "Allow Drop types in statics/const functions".

This commit is contained in:
Eduard Burtescu 2016-05-05 08:11:28 +03:00
parent 4f5900aefa
commit ed66fe48e9
6 changed files with 54 additions and 23 deletions

View File

@ -345,7 +345,7 @@ fn drop(&mut self) {}
const F : Foo = Foo { a : 0 };
// error: constants are not allowed to have destructors
static S : Foo = Foo { a : 0 };
// error: statics are not allowed to have destructors
// error: destructors in statics are an unstable feature
```
To solve this issue, please use a type which does allow the usage of type with

View File

@ -222,13 +222,40 @@ fn nest<F: FnOnce(&mut Self)>(&mut self, f: F) {
}
/// Check for NEEDS_DROP (from an ADT or const fn call) and
/// error, unless we're in a function.
/// error, unless we're in a function, or the feature-gate
/// for globals with destructors is enabled.
fn deny_drop(&self) {
if self.mode != Mode::Fn && self.qualif.intersects(Qualif::NEEDS_DROP) {
span_err!(self.tcx.sess, self.span, E0493,
"{}s are not allowed to have destructors",
self.mode);
if self.mode == Mode::Fn || !self.qualif.intersects(Qualif::NEEDS_DROP) {
return;
}
// Static and const fn's allow destructors, but they're feature-gated.
let msg = if self.mode != Mode::Const {
// Feature-gate for globals with destructors is enabled.
if self.tcx.sess.features.borrow().drop_types_in_const {
return;
}
// This comes from a macro that has #[allow_internal_unstable].
if self.tcx.sess.codemap().span_allows_unstable(self.span) {
return;
}
format!("destructors in {}s are an unstable feature",
self.mode)
} else {
format!("{}s are not allowed to have destructors",
self.mode)
};
let mut err =
struct_span_err!(self.tcx.sess, self.span, E0493, "{}", msg);
if self.mode != Mode::Const {
help!(&mut err,
"in Nightly builds, add `#![feature(drop_types_in_const)]` \
to the crate attributes to enable");
}
err.emit();
}
/// Check if an Lvalue with the current qualifications could

View File

@ -266,7 +266,10 @@ pub fn new() -> Features {
(active, specialization, "1.7.0", Some(31844)),
// pub(restricted) visibilities (RFC 1422)
(active, pub_restricted, "1.9.0", Some(32409))
(active, pub_restricted, "1.9.0", Some(32409)),
// Allow Drop types in statics/const functions (RFC 1440)
(active, drop_types_in_const, "1.9.0", Some(33156))
);
declare_features! (

View File

@ -37,7 +37,7 @@ enum SafeEnum {
// This one should fail
static STATIC3: SafeEnum = SafeEnum::Variant3(WithDtor);
//~^ ERROR statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
// This enum will be used to test that variants
@ -54,9 +54,9 @@ fn drop(&mut self) {}
static STATIC4: UnsafeEnum = UnsafeEnum::Variant5;
//~^ ERROR statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
static STATIC5: UnsafeEnum = UnsafeEnum::Variant6(0);
//~^ ERROR statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
struct SafeStruct {
@ -71,7 +71,7 @@ struct SafeStruct {
// field2 has an unsafe value, hence this should fail
static STATIC7: SafeStruct = SafeStruct{field1: SafeEnum::Variant1,
field2: SafeEnum::Variant3(WithDtor)};
//~^ ERROR statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
// Test variadic constructor for structs. The base struct should be examined
// as well as every field present in the constructor.
@ -84,7 +84,7 @@ struct SafeStruct {
static STATIC9: SafeStruct = SafeStruct{field1: SafeEnum::Variant1,
..SafeStruct{field1: SafeEnum::Variant3(WithDtor),
field2: SafeEnum::Variant1}};
//~^^ ERROR statics are not allowed to have destructors
//~^^ ERROR destructors in statics are an unstable feature
struct UnsafeStruct;
@ -94,7 +94,7 @@ fn drop(&mut self) {}
// Types with destructors are not allowed for statics
static STATIC10: UnsafeStruct = UnsafeStruct;
//~^ ERROR statics are not allowed to have destructor
//~^ ERROR destructors in statics are an unstable feature
struct MyOwned;
@ -105,19 +105,19 @@ fn drop(&mut self) {}
// to have types with destructors
// These should fail
static mut STATIC12: UnsafeStruct = UnsafeStruct;
//~^ ERROR mutable statics are not allowed to have destructors
//~^^ ERROR statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
//~^^ ERROR destructors in statics are an unstable feature
static mut STATIC13: SafeStruct = SafeStruct{field1: SafeEnum::Variant1,
//~^ ERROR mutable statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
field2: SafeEnum::Variant3(WithDtor)};
//~^ ERROR: statics are not allowed to have destructors
//~^ ERROR: destructors in statics are an unstable feature
static mut STATIC14: SafeStruct = SafeStruct {
//~^ ERROR mutable statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
field1: SafeEnum::Variant1,
field2: SafeEnum::Variant4("str".to_string())
//~^ ERROR method calls in statics are limited to constant inherent methods
//~^ ERROR calls in statics are limited to constant functions
};
static STATIC15: &'static [Box<MyOwned>] = &[
@ -131,7 +131,7 @@ fn drop(&mut self) {}
);
static mut STATIC17: SafeEnum = SafeEnum::Variant1;
//~^ ERROR mutable statics are not allowed to have destructors
//~^ ERROR destructors in statics are an unstable feature
static STATIC19: Box<isize> =
box 3;

View File

@ -10,11 +10,12 @@
// Regression test for issue 9243
struct Test {
pub struct Test {
mem: isize,
}
pub static g_test: Test = Test {mem: 0}; //~ ERROR statics are not allowed to have destructors
pub static g_test: Test = Test {mem: 0};
//~^ ERROR destructors in statics are an unstable feature
impl Drop for Test {
fn drop(&mut self) {}

View File

@ -12,6 +12,6 @@
static mut a: Box<isize> = box 3;
//~^ ERROR allocations are not allowed in statics
//~^^ ERROR mutable statics are not allowed to have boxes
//~^^ ERROR destructors in statics are an unstable feature
fn main() {}