summary refs log tree commit diff
path: root/src/test/ui/mir-dataflow/uninits-1.rs
blob: 66b3f458a5159f932f0e033fc358ee86d8b71995 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
// General test of maybe_uninits state computed by MIR dataflow.

#![feature(core_intrinsics, rustc_attrs)]

use std::intrinsics::rustc_peek;
use std::mem::{drop, replace};

struct S(i32);

#[rustc_mir(rustc_peek_maybe_uninit,stop_after_dataflow)]
fn foo(test: bool, x: &mut S, y: S, mut z: S) -> S {
    let ret;
    // `ret` starts off uninitialized
    unsafe { rustc_peek(&ret); }

    // All function formal parameters start off initialized.

    unsafe { rustc_peek(&x) }; //~ ERROR rustc_peek: bit not set
    unsafe { rustc_peek(&y) }; //~ ERROR rustc_peek: bit not set
    unsafe { rustc_peek(&z) }; //~ ERROR rustc_peek: bit not set

    ret = if test {
        ::std::mem::replace(x, y)
    } else {
        z = y;
        z
    };

    // `z` may be uninitialized here.
    unsafe { rustc_peek(&z); }

    // `y` is definitely uninitialized here.
    unsafe { rustc_peek(&y); }

    // `x` is still (definitely) initialized (replace above is a reborrow).
    unsafe { rustc_peek(&x); } //~ ERROR rustc_peek: bit not set

    ::std::mem::drop(x);

    // `x` is *definitely* uninitialized here
    unsafe { rustc_peek(&x); }

    // `ret` is now definitely initialized (via `if` above).
    unsafe { rustc_peek(&ret); } //~ ERROR rustc_peek: bit not set

    ret
}
fn main() {
    foo(true, &mut S(13), S(14), S(15));
    foo(false, &mut S(13), S(14), S(15));
}