@@ -18,6 +18,7 @@ mod unsafe_list;
18
18
19
19
use crate :: num:: NonZeroUsize ;
20
20
use crate :: ops:: { Deref , DerefMut } ;
21
+ use crate :: panic:: { self , AssertUnwindSafe } ;
21
22
use crate :: time:: Duration ;
22
23
23
24
use super :: abi:: thread;
@@ -147,7 +148,8 @@ impl WaitQueue {
147
148
/// Adds the calling thread to the `WaitVariable`'s wait queue, then wait
148
149
/// until a wakeup event.
149
150
///
150
- /// This function does not return until this thread has been awoken.
151
+ /// This function does not return until this thread has been awoken. When `before_wait` panics,
152
+ /// this function will abort.
151
153
pub fn wait < T , F : FnOnce ( ) > ( mut guard : SpinMutexGuard < ' _ , WaitVariable < T > > , before_wait : F ) {
152
154
// very unsafe: check requirements of UnsafeList::push
153
155
unsafe {
@@ -157,8 +159,13 @@ impl WaitQueue {
157
159
} ) ) ;
158
160
let entry = guard. queue . inner . push ( & mut entry) ;
159
161
drop ( guard) ;
160
- before_wait ( ) ;
162
+ if let Err ( _e) = panic:: catch_unwind ( AssertUnwindSafe ( || before_wait ( ) ) ) {
163
+ rtabort ! ( "Panic before wait on wakeup event" )
164
+ }
161
165
while !entry. lock ( ) . wake {
166
+ // `entry.wake` is only set in `notify_one` and `notify_all` functions. Both ensure
167
+ // the entry is removed from the queue _before_ setting this bool. There are no
168
+ // other references to `entry`.
162
169
// don't panic, this would invalidate `entry` during unwinding
163
170
let eventset = rtunwrap ! ( Ok , usercalls:: wait( EV_UNPARK , WAIT_INDEFINITE ) ) ;
164
171
rtassert ! ( eventset & EV_UNPARK == EV_UNPARK ) ;
@@ -169,6 +176,7 @@ impl WaitQueue {
169
176
/// Adds the calling thread to the `WaitVariable`'s wait queue, then wait
170
177
/// until a wakeup event or timeout. If event was observed, returns true.
171
178
/// If not, it will remove the calling thread from the wait queue.
179
+ /// When `before_wait` panics, this function will abort.
172
180
pub fn wait_timeout < T , F : FnOnce ( ) > (
173
181
lock : & SpinMutex < WaitVariable < T > > ,
174
182
timeout : Duration ,
@@ -181,9 +189,13 @@ impl WaitQueue {
181
189
wake : false ,
182
190
} ) ) ;
183
191
let entry_lock = lock. lock ( ) . queue . inner . push ( & mut entry) ;
184
- before_wait ( ) ;
192
+ if let Err ( _e) = panic:: catch_unwind ( AssertUnwindSafe ( || before_wait ( ) ) ) {
193
+ rtabort ! ( "Panic before wait on wakeup event or timeout" )
194
+ }
185
195
usercalls:: wait_timeout ( EV_UNPARK , timeout, || entry_lock. lock ( ) . wake ) ;
186
- // acquire the wait queue's lock first to avoid deadlock.
196
+ // acquire the wait queue's lock first to avoid deadlock
197
+ // and ensure no other function can simultaneously access the list
198
+ // (e.g., `notify_one` or `notify_all`)
187
199
let mut guard = lock. lock ( ) ;
188
200
let success = entry_lock. lock ( ) . wake ;
189
201
if !success {
@@ -204,8 +216,8 @@ impl WaitQueue {
204
216
) -> Result < WaitGuard < ' _ , T > , SpinMutexGuard < ' _ , WaitVariable < T > > > {
205
217
// SAFETY: lifetime of the pop() return value is limited to the map
206
218
// closure (The closure return value is 'static). The underlying
207
- // stack frame won't be freed until after the WaitGuard created below
208
- // is dropped.
219
+ // stack frame won't be freed until after the lock on the queue is released
220
+ // (i.e., `guard` is dropped) .
209
221
unsafe {
210
222
let tcs = guard. queue . inner . pop ( ) . map ( |entry| -> Tcs {
211
223
let mut entry_guard = entry. lock ( ) ;
@@ -231,7 +243,7 @@ impl WaitQueue {
231
243
) -> Result < WaitGuard < ' _ , T > , SpinMutexGuard < ' _ , WaitVariable < T > > > {
232
244
// SAFETY: lifetime of the pop() return values are limited to the
233
245
// while loop body. The underlying stack frames won't be freed until
234
- // after the WaitGuard created below is dropped.
246
+ // after the lock on the queue is released (i.e., `guard` is dropped) .
235
247
unsafe {
236
248
let mut count = 0 ;
237
249
while let Some ( entry) = guard. queue . inner . pop ( ) {
0 commit comments