@@ -20,24 +20,29 @@ use crate::lsps0;
20
20
use crate :: lsps1;
21
21
use crate :: lsps2;
22
22
use crate :: prelude:: { Vec , VecDeque } ;
23
- use crate :: sync:: Mutex ;
23
+ use crate :: sync:: { Arc , Mutex } ;
24
+
25
+ use core:: future:: Future ;
26
+ use core:: task:: { Poll , Waker } ;
24
27
25
28
pub ( crate ) struct EventQueue {
26
- queue : Mutex < VecDeque < Event > > ,
29
+ queue : Arc < Mutex < VecDeque < Event > > > ,
30
+ waker : Arc < Mutex < Option < Waker > > > ,
27
31
#[ cfg( feature = "std" ) ]
28
32
condvar : std:: sync:: Condvar ,
29
33
}
30
34
31
35
impl EventQueue {
32
36
pub fn new ( ) -> Self {
33
- let queue = Mutex :: new ( VecDeque :: new ( ) ) ;
37
+ let queue = Arc :: new ( Mutex :: new ( VecDeque :: new ( ) ) ) ;
38
+ let waker = Arc :: new ( Mutex :: new ( None ) ) ;
34
39
#[ cfg( feature = "std" ) ]
35
40
{
36
41
let condvar = std:: sync:: Condvar :: new ( ) ;
37
- Self { queue, condvar }
42
+ Self { queue, waker , condvar }
38
43
}
39
44
#[ cfg( not( feature = "std" ) ) ]
40
- Self { queue }
45
+ Self { queue, waker }
41
46
}
42
47
43
48
pub fn enqueue ( & self , event : Event ) {
@@ -46,6 +51,9 @@ impl EventQueue {
46
51
queue. push_back ( event) ;
47
52
}
48
53
54
+ if let Some ( waker) = self . waker . lock ( ) . unwrap ( ) . take ( ) {
55
+ waker. wake ( ) ;
56
+ }
49
57
#[ cfg( feature = "std" ) ]
50
58
self . condvar . notify_one ( ) ;
51
59
}
@@ -54,6 +62,10 @@ impl EventQueue {
54
62
self . queue . lock ( ) . unwrap ( ) . pop_front ( )
55
63
}
56
64
65
+ pub async fn next_event_async ( & self ) -> Event {
66
+ EventFuture { event_queue : Arc :: clone ( & self . queue ) , waker : Arc :: clone ( & self . waker ) } . await
67
+ }
68
+
57
69
#[ cfg( feature = "std" ) ]
58
70
pub fn wait_next_event ( & self ) -> Event {
59
71
let mut queue =
@@ -65,6 +77,10 @@ impl EventQueue {
65
77
drop ( queue) ;
66
78
67
79
if should_notify {
80
+ if let Some ( waker) = self . waker . lock ( ) . unwrap ( ) . take ( ) {
81
+ waker. wake ( ) ;
82
+ }
83
+
68
84
self . condvar . notify_one ( ) ;
69
85
}
70
86
@@ -92,3 +108,23 @@ pub enum Event {
92
108
/// An LSPS2 (JIT Channel) server event.
93
109
LSPS2Service ( lsps2:: event:: LSPS2ServiceEvent ) ,
94
110
}
111
+
112
+ struct EventFuture {
113
+ event_queue : Arc < Mutex < VecDeque < Event > > > ,
114
+ waker : Arc < Mutex < Option < Waker > > > ,
115
+ }
116
+
117
+ impl Future for EventFuture {
118
+ type Output = Event ;
119
+
120
+ fn poll (
121
+ self : core:: pin:: Pin < & mut Self > , cx : & mut core:: task:: Context < ' _ > ,
122
+ ) -> core:: task:: Poll < Self :: Output > {
123
+ if let Some ( event) = self . event_queue . lock ( ) . unwrap ( ) . pop_front ( ) {
124
+ Poll :: Ready ( event)
125
+ } else {
126
+ * self . waker . lock ( ) . unwrap ( ) = Some ( cx. waker ( ) . clone ( ) ) ;
127
+ Poll :: Pending
128
+ }
129
+ }
130
+ }
0 commit comments