78da577650
Cleanup blocks are declared with `bb (cleanup) = { ... }`. `Call` and `Drop` terminators take an additional argument describing the unwind action, which is one of the following: * `UnwindContinue()` * `UnwindUnreachable()` * `UnwindTerminate(reason)`, where reason is `ReasonAbi` or `ReasonInCleanup` * `UnwindCleanup(block)` Also support unwind resume and unwind terminate terminators: * `UnwindResume()` * `UnwindTerminate(reason)`
44 lines
1.0 KiB
Rust
44 lines
1.0 KiB
Rust
// skip-filecheck
|
|
// This example is interesting because the non-transitive version of `MaybeLiveLocals` would
|
|
// report that *all* of these stores are live.
|
|
//
|
|
// needs-unwind
|
|
// unit-test: DeadStoreElimination
|
|
|
|
#![feature(core_intrinsics, custom_mir)]
|
|
use std::intrinsics::mir::*;
|
|
|
|
#[inline(never)]
|
|
fn cond() -> bool {
|
|
false
|
|
}
|
|
|
|
// EMIT_MIR cycle.cycle.DeadStoreElimination.diff
|
|
#[custom_mir(dialect = "runtime", phase = "post-cleanup")]
|
|
fn cycle(mut x: i32, mut y: i32, mut z: i32) {
|
|
// We use custom MIR to avoid generating debuginfo, that would force to preserve writes.
|
|
mir!(
|
|
let condition: bool;
|
|
{
|
|
Call(condition = cond(), bb1, UnwindContinue())
|
|
}
|
|
bb1 = {
|
|
match condition { true => bb2, _ => ret }
|
|
}
|
|
bb2 = {
|
|
let temp = z;
|
|
z = y;
|
|
y = x;
|
|
x = temp;
|
|
Call(condition = cond(), bb1, UnwindContinue())
|
|
}
|
|
ret = {
|
|
Return()
|
|
}
|
|
)
|
|
}
|
|
|
|
fn main() {
|
|
cycle(1, 2, 3);
|
|
}
|