1 | use futures::task::{self, ArcWake, Waker}; |
2 | use std::panic; |
3 | use std::sync::{Arc, Mutex}; |
4 | |
5 | struct CountingWaker { |
6 | nr_wake: Mutex<i32>, |
7 | } |
8 | |
9 | impl CountingWaker { |
10 | fn new() -> Self { |
11 | Self { nr_wake: Mutex::new(0) } |
12 | } |
13 | |
14 | fn wakes(&self) -> i32 { |
15 | *self.nr_wake.lock().unwrap() |
16 | } |
17 | } |
18 | |
19 | impl ArcWake for CountingWaker { |
20 | fn wake_by_ref(arc_self: &Arc<Self>) { |
21 | let mut lock = arc_self.nr_wake.lock().unwrap(); |
22 | *lock += 1; |
23 | } |
24 | } |
25 | |
26 | #[test] |
27 | fn create_from_arc() { |
28 | let some_w = Arc::new(CountingWaker::new()); |
29 | |
30 | let w1: Waker = task::waker(some_w.clone()); |
31 | assert_eq!(2, Arc::strong_count(&some_w)); |
32 | w1.wake_by_ref(); |
33 | assert_eq!(1, some_w.wakes()); |
34 | |
35 | let w2 = w1.clone(); |
36 | assert_eq!(3, Arc::strong_count(&some_w)); |
37 | |
38 | w2.wake_by_ref(); |
39 | assert_eq!(2, some_w.wakes()); |
40 | |
41 | drop(w2); |
42 | assert_eq!(2, Arc::strong_count(&some_w)); |
43 | drop(w1); |
44 | assert_eq!(1, Arc::strong_count(&some_w)); |
45 | } |
46 | |
47 | #[test] |
48 | fn ref_wake_same() { |
49 | let some_w = Arc::new(CountingWaker::new()); |
50 | |
51 | let w1: Waker = task::waker(some_w.clone()); |
52 | let w2 = task::waker_ref(&some_w); |
53 | let w3 = w2.clone(); |
54 | |
55 | assert!(w1.will_wake(&w2)); |
56 | assert!(w2.will_wake(&w3)); |
57 | } |
58 | |
59 | #[test] |
60 | fn proper_refcount_on_wake_panic() { |
61 | struct PanicWaker; |
62 | |
63 | impl ArcWake for PanicWaker { |
64 | fn wake_by_ref(_arc_self: &Arc<Self>) { |
65 | panic!("WAKE UP" ); |
66 | } |
67 | } |
68 | |
69 | let some_w = Arc::new(PanicWaker); |
70 | |
71 | let w1: Waker = task::waker(some_w.clone()); |
72 | assert_eq!( |
73 | "WAKE UP" , |
74 | *panic::catch_unwind(|| w1.wake_by_ref()).unwrap_err().downcast::<&str>().unwrap() |
75 | ); |
76 | assert_eq!(2, Arc::strong_count(&some_w)); // some_w + w1 |
77 | drop(w1); |
78 | assert_eq!(1, Arc::strong_count(&some_w)); // some_w |
79 | } |
80 | |