summaryrefslogtreecommitdiffstats
path: root/tests/coverage/async.coverage
blob: 07bc16c2d926c26bc4c33c6cba043f47a7dce68d (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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
   LL|       |#![allow(unused_assignments, dead_code)]
   LL|       |
   LL|       |// compile-flags: --edition=2018 -C opt-level=1
   LL|       |
   LL|      1|async fn c(x: u8) -> u8 {
   LL|      1|    if x == 8 {
   LL|      1|        1
   LL|       |    } else {
   LL|      0|        0
   LL|       |    }
   LL|      1|}
   LL|       |
   LL|      0|async fn d() -> u8 { 1 }
   LL|       |
   LL|      0|async fn e() -> u8 { 1 } // unused function; executor does not block on `g()`
   LL|       |
   LL|      1|async fn f() -> u8 { 1 }
   LL|       |
   LL|      0|async fn foo() -> [bool; 10] { [false; 10] } // unused function; executor does not block on `h()`
   LL|       |
   LL|      1|pub async fn g(x: u8) {
   LL|      0|    match x {
   LL|      0|        y if e().await == y => (),
   LL|      0|        y if f().await == y => (),
   LL|      0|        _ => (),
   LL|       |    }
   LL|      0|}
   LL|       |
   LL|      1|async fn h(x: usize) { // The function signature is counted when called, but the body is not
   LL|      0|                       // executed (not awaited) so the open brace has a `0` count (at least when
   LL|      0|                       // displayed with `llvm-cov show` in color-mode).
   LL|      0|    match x {
   LL|      0|        y if foo().await[y] => (),
   LL|      0|        _ => (),
   LL|       |    }
   LL|      0|}
   LL|       |
   LL|      1|async fn i(x: u8) { // line coverage is 1, but there are 2 regions:
   LL|      1|                    // (a) the function signature, counted when the function is called; and
   LL|      1|                    // (b) the open brace for the function body, counted once when the body is
   LL|      1|                    // executed asynchronously.
   LL|      1|    match x {
   LL|      1|        y if c(x).await == y + 1 => { d().await; }
                      ^0        ^0                  ^0  ^0
   LL|      1|        y if f().await == y + 1 => (),
                      ^0       ^0                ^0
   LL|      1|        _ => (),
   LL|       |    }
   LL|      1|}
   LL|       |
   LL|      1|fn j(x: u8) {
   LL|      1|    // non-async versions of `c()`, `d()`, and `f()` to make it similar to async `i()`.
   LL|      1|    fn c(x: u8) -> u8 {
   LL|      1|        if x == 8 {
   LL|      1|            1 // This line appears covered, but the 1-character expression span covering the `1`
                          ^0
   LL|      1|              // is not executed. (`llvm-cov show` displays a `^0` below the `1` ). This is because
   LL|      1|              // `fn j()` executes the open brace for the function body, followed by the function's
   LL|      1|              // first executable statement, `match x`. Inner function declarations are not
   LL|      1|              // "visible" to the MIR for `j()`, so the code region counts all lines between the
   LL|      1|              // open brace and the first statement as executed, which is, in a sense, true.
   LL|      1|              // `llvm-cov show` overcomes this kind of situation by showing the actual counts
   LL|      1|              // of the enclosed coverages, (that is, the `1` expression was not executed, and
   LL|      1|              // accurately displays a `0`).
   LL|      1|        } else {
   LL|      1|            0
   LL|      1|        }
   LL|      1|    }
   LL|      1|    fn d() -> u8 { 1 } // inner function is defined in-line, but the function is not executed
                  ^0
   LL|      1|    fn f() -> u8 { 1 }
   LL|      1|    match x {
   LL|      1|        y if c(x) == y + 1 => { d(); }
                      ^0                    ^0
   LL|      1|        y if f() == y + 1 => (),
                      ^0                   ^0
   LL|      1|        _ => (),
   LL|       |    }
   LL|      1|}
   LL|       |
   LL|      0|fn k(x: u8) { // unused function
   LL|      0|    match x {
   LL|      0|        1 => (),
   LL|      0|        2 => (),
   LL|      0|        _ => (),
   LL|       |    }
   LL|      0|}
   LL|       |
   LL|      1|fn l(x: u8) {
   LL|      1|    match x {
   LL|      0|        1 => (),
   LL|      0|        2 => (),
   LL|      1|        _ => (),
   LL|       |    }
   LL|      1|}
   LL|       |
   LL|      1|async fn m(x: u8) -> u8 { x - 1 }
                                      ^0
   LL|       |
   LL|      1|fn main() {
   LL|      1|    let _ = g(10);
   LL|      1|    let _ = h(9);
   LL|      1|    let mut future = Box::pin(i(8));
   LL|      1|    j(7);
   LL|      1|    l(6);
   LL|      1|    let _ = m(5);
   LL|      1|    executor::block_on(future.as_mut());
   LL|      1|}
   LL|       |
   LL|       |mod executor {
   LL|       |    use core::{
   LL|       |        future::Future,
   LL|       |        pin::Pin,
   LL|       |        task::{Context, Poll, RawWaker, RawWakerVTable, Waker},
   LL|       |    };
   LL|       |
   LL|      1|    pub fn block_on<F: Future>(mut future: F) -> F::Output {
   LL|      1|        let mut future = unsafe { Pin::new_unchecked(&mut future) };
   LL|      1|        use std::hint::unreachable_unchecked;
   LL|      1|        static VTABLE: RawWakerVTable = RawWakerVTable::new(
   LL|      1|            |_| unsafe { unreachable_unchecked() }, // clone
                              ^0
   LL|      1|            |_| unsafe { unreachable_unchecked() }, // wake
                              ^0
   LL|      1|            |_| unsafe { unreachable_unchecked() }, // wake_by_ref
                              ^0
   LL|      1|            |_| (),
   LL|      1|        );
   LL|      1|        let waker = unsafe { Waker::from_raw(RawWaker::new(core::ptr::null(), &VTABLE)) };
   LL|      1|        let mut context = Context::from_waker(&waker);
   LL|       |
   LL|       |        loop {
   LL|      1|            if let Poll::Ready(val) = future.as_mut().poll(&mut context) {
   LL|      1|                break val;
   LL|      0|            }
   LL|       |        }
   LL|      1|    }
   LL|       |}