-
Notifications
You must be signed in to change notification settings - Fork 12.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Modify MIR building to drop
foo
in [foo; 0]
- Loading branch information
1 parent
222c572
commit 0f65bcd
Showing
5 changed files
with
210 additions
and
5 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
fn borrowck_catch() { | ||
let foo = String::new(); | ||
let _bar = foo; | ||
let _baz = [foo; 0]; //~ ERROR use of moved value: `foo` [E0382] | ||
} | ||
|
||
const _: [String; 0] = [String::new(); 0]; | ||
//~^ ERROR destructors cannot be evaluated at compile-time [E0493] | ||
|
||
fn must_be_init() { | ||
let x: u8; | ||
let _ = [x; 0]; //~ ERROR: use of possibly-uninitialized variable: `x` | ||
} | ||
|
||
fn main() {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
error[E0382]: use of moved value: `foo` | ||
--> $DIR/repeat-drop-2.rs:4:17 | ||
| | ||
LL | let foo = String::new(); | ||
| --- move occurs because `foo` has type `String`, which does not implement the `Copy` trait | ||
LL | let _bar = foo; | ||
| --- value moved here | ||
LL | let _baz = [foo; 0]; | ||
| ^^^ value used here after move | ||
|
||
error[E0493]: destructors cannot be evaluated at compile-time | ||
--> $DIR/repeat-drop-2.rs:7:25 | ||
| | ||
LL | const _: [String; 0] = [String::new(); 0]; | ||
| -^^^^^^^^^^^^^---- | ||
| || | ||
| |constants cannot evaluate destructors | ||
| value is dropped here | ||
|
||
error[E0381]: use of possibly-uninitialized variable: `x` | ||
--> $DIR/repeat-drop-2.rs:12:14 | ||
| | ||
LL | let _ = [x; 0]; | ||
| ^ use of possibly-uninitialized `x` | ||
|
||
error: aborting due to 3 previous errors | ||
|
||
Some errors have detailed explanations: E0381, E0382, E0493. | ||
For more information about an error, try `rustc --explain E0381`. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,120 @@ | ||
// run-pass | ||
// ignore-wasm32-bare no unwinding panic | ||
// ignore-avr no unwinding panic | ||
// ignore-nvptx64 no unwinding panic | ||
|
||
static mut CHECK: usize = 0; | ||
|
||
struct DropChecker(usize); | ||
|
||
impl Drop for DropChecker { | ||
fn drop(&mut self) { | ||
unsafe { | ||
if CHECK != self.0 - 1 { | ||
panic!("Found {}, should have found {}", CHECK, self.0 - 1); | ||
} | ||
CHECK = self.0; | ||
} | ||
} | ||
} | ||
|
||
macro_rules! check_drops { | ||
($l:literal) => { | ||
unsafe { assert_eq!(CHECK, $l) } | ||
}; | ||
} | ||
|
||
struct DropPanic; | ||
|
||
impl Drop for DropPanic { | ||
fn drop(&mut self) { | ||
panic!() | ||
} | ||
} | ||
|
||
fn value_zero() { | ||
unsafe { CHECK = 0 }; | ||
let foo = DropChecker(1); | ||
let v: [DropChecker; 0] = [foo; 0]; | ||
check_drops!(1); | ||
std::mem::drop(v); | ||
check_drops!(1); | ||
} | ||
|
||
fn value_one() { | ||
unsafe { CHECK = 0 }; | ||
let foo = DropChecker(1); | ||
let v: [DropChecker; 1] = [foo; 1]; | ||
check_drops!(0); | ||
std::mem::drop(v); | ||
check_drops!(1); | ||
} | ||
|
||
const DROP_CHECKER: DropChecker = DropChecker(1); | ||
|
||
fn const_zero() { | ||
unsafe { CHECK = 0 }; | ||
let v: [DropChecker; 0] = [DROP_CHECKER; 0]; | ||
check_drops!(0); | ||
std::mem::drop(v); | ||
check_drops!(0); | ||
} | ||
|
||
fn const_one() { | ||
unsafe { CHECK = 0 }; | ||
let v: [DropChecker; 1] = [DROP_CHECKER; 1]; | ||
check_drops!(0); | ||
std::mem::drop(v); | ||
check_drops!(1); | ||
} | ||
|
||
fn const_generic_zero<const N: usize>() { | ||
unsafe { CHECK = 0 }; | ||
let v: [DropChecker; N] = [DROP_CHECKER; N]; | ||
check_drops!(0); | ||
std::mem::drop(v); | ||
check_drops!(0); | ||
} | ||
|
||
fn const_generic_one<const N: usize>() { | ||
unsafe { CHECK = 0 }; | ||
let v: [DropChecker; N] = [DROP_CHECKER; N]; | ||
check_drops!(0); | ||
std::mem::drop(v); | ||
check_drops!(1); | ||
} | ||
|
||
// Make sure that things are allowed to promote as expected | ||
|
||
fn allow_promote() { | ||
unsafe { CHECK = 0 }; | ||
let foo = DropChecker(1); | ||
let v: &'static [DropChecker; 0] = &[foo; 0]; | ||
check_drops!(1); | ||
std::mem::drop(v); | ||
check_drops!(1); | ||
} | ||
|
||
// Verify that unwinding in the drop causes the right things to drop in the right order | ||
fn on_unwind() { | ||
unsafe { CHECK = 0 }; | ||
std::panic::catch_unwind(|| { | ||
let panic = DropPanic; | ||
let _local = DropChecker(2); | ||
let _v = (DropChecker(1), [panic; 0]); | ||
std::process::abort(); | ||
}) | ||
.unwrap_err(); | ||
check_drops!(2); | ||
} | ||
|
||
fn main() { | ||
value_zero(); | ||
value_one(); | ||
const_zero(); | ||
const_one(); | ||
const_generic_zero::<0>(); | ||
const_generic_one::<1>(); | ||
allow_promote(); | ||
on_unwind(); | ||
} |