@@ -223,7 +223,9 @@ struct Waiter {
223
223
/// `Notify`, or it is exclusively owned by the enclosing `Waiter`.
224
224
waker : UnsafeCell < Option < Waker > > ,
225
225
226
- /// Notification for this waiter.
226
+ /// Notification for this waiter. Uses 2 bits to store if and how was
227
+ /// notified, 1 bit for storing if it was woken up using FIFO or LIFO, and
228
+ /// the rest of it is unused.
227
229
/// * if it's `None`, then `waker` is protected by the `waiters` lock.
228
230
/// * if it's `Some`, then `waker` is exclusively owned by the
229
231
/// enclosing `Waiter` and can be accessed without locking.
@@ -253,13 +255,16 @@ generate_addr_of_methods! {
253
255
}
254
256
255
257
// No notification.
256
- const NOTIFICATION_NONE : usize = 0 ;
258
+ const NOTIFICATION_NONE : usize = 0b000 ;
257
259
258
260
// Notification type used by `notify_one`.
259
- const NOTIFICATION_ONE : usize = 1 ;
261
+ const NOTIFICATION_ONE : usize = 0b001 ;
262
+
263
+ // Notification type used by `notify_last`.
264
+ const NOTIFICATION_LAST : usize = 0b101 ;
260
265
261
266
// Notification type used by `notify_waiters`.
262
- const NOTIFICATION_ALL : usize = 2 ;
267
+ const NOTIFICATION_ALL : usize = 0b010 ;
263
268
264
269
/// Notification for a `Waiter`.
265
270
/// This struct is equivalent to `Option<Notification>`, but uses
@@ -275,13 +280,20 @@ impl AtomicNotification {
275
280
/// Store-release a notification.
276
281
/// This method should be called exactly once.
277
282
fn store_release ( & self , notification : Notification ) {
278
- self . 0 . store ( notification as usize , Release ) ;
283
+ let data: usize = match notification {
284
+ Notification :: All => NOTIFICATION_ALL ,
285
+ Notification :: One ( NotifyOneStrategy :: Fifo ) => NOTIFICATION_ONE ,
286
+ Notification :: One ( NotifyOneStrategy :: Lifo ) => NOTIFICATION_LAST ,
287
+ } ;
288
+ self . 0 . store ( data, Release ) ;
279
289
}
280
290
281
291
fn load ( & self , ordering : Ordering ) -> Option < Notification > {
282
- match self . 0 . load ( ordering) {
292
+ let data = self . 0 . load ( ordering) ;
293
+ match data {
283
294
NOTIFICATION_NONE => None ,
284
- NOTIFICATION_ONE => Some ( Notification :: One ) ,
295
+ NOTIFICATION_ONE => Some ( Notification :: One ( NotifyOneStrategy :: Fifo ) ) ,
296
+ NOTIFICATION_LAST => Some ( Notification :: One ( NotifyOneStrategy :: Lifo ) ) ,
285
297
NOTIFICATION_ALL => Some ( Notification :: All ) ,
286
298
_ => unreachable ! ( ) ,
287
299
}
@@ -296,11 +308,18 @@ impl AtomicNotification {
296
308
}
297
309
}
298
310
311
+ #[ derive( Debug , PartialEq , Eq ) ]
312
+ #[ repr( usize ) ]
313
+ enum NotifyOneStrategy {
314
+ Fifo ,
315
+ Lifo ,
316
+ }
317
+
299
318
#[ derive( Debug , PartialEq , Eq ) ]
300
319
#[ repr( usize ) ]
301
320
enum Notification {
302
- One = NOTIFICATION_ONE ,
303
- All = NOTIFICATION_ALL ,
321
+ One ( NotifyOneStrategy ) ,
322
+ All ,
304
323
}
305
324
306
325
/// List used in `Notify::notify_waiters`. It wraps a guarded linked list
@@ -521,7 +540,7 @@ impl Notify {
521
540
}
522
541
}
523
542
524
- /// Notifies a waiting task.
543
+ /// Notifies the first waiting task.
525
544
///
526
545
/// If a task is currently waiting, that task is notified. Otherwise, a
527
546
/// permit is stored in this `Notify` value and the **next** call to
@@ -558,6 +577,23 @@ impl Notify {
558
577
// Alias for old name in 0.x
559
578
#[ cfg_attr( docsrs, doc( alias = "notify" ) ) ]
560
579
pub fn notify_one ( & self ) {
580
+ self . notify_with_strategy ( NotifyOneStrategy :: Fifo ) ;
581
+ }
582
+
583
+ /// Notifies the last waiting task.
584
+ ///
585
+ /// This function behaves similar to `notify_one`. The only difference is that it wakes
586
+ /// the most recently added waiter instead of the oldest waiter.
587
+ ///
588
+ /// Check the [`notify_one()`] documentation for more info and
589
+ /// examples.
590
+ ///
591
+ /// [`notify_one()`]: Notify::notify_one
592
+ pub fn notify_last ( & self ) {
593
+ self . notify_with_strategy ( NotifyOneStrategy :: Lifo ) ;
594
+ }
595
+
596
+ fn notify_with_strategy ( & self , strategy : NotifyOneStrategy ) {
561
597
// Load the current state
562
598
let mut curr = self . state . load ( SeqCst ) ;
563
599
@@ -585,7 +621,7 @@ impl Notify {
585
621
// transition out of WAITING while the lock is held.
586
622
curr = self . state . load ( SeqCst ) ;
587
623
588
- if let Some ( waker) = notify_locked ( & mut waiters, & self . state , curr) {
624
+ if let Some ( waker) = notify_locked ( & mut waiters, & self . state , curr, strategy ) {
589
625
drop ( waiters) ;
590
626
waker. wake ( ) ;
591
627
}
@@ -708,7 +744,12 @@ impl Default for Notify {
708
744
impl UnwindSafe for Notify { }
709
745
impl RefUnwindSafe for Notify { }
710
746
711
- fn notify_locked ( waiters : & mut WaitList , state : & AtomicUsize , curr : usize ) -> Option < Waker > {
747
+ fn notify_locked (
748
+ waiters : & mut WaitList ,
749
+ state : & AtomicUsize ,
750
+ curr : usize ,
751
+ strategy : NotifyOneStrategy ,
752
+ ) -> Option < Waker > {
712
753
match get_state ( curr) {
713
754
EMPTY | NOTIFIED => {
714
755
let res = state. compare_exchange ( curr, set_state ( curr, NOTIFIED ) , SeqCst , SeqCst ) ;
@@ -728,8 +769,11 @@ fn notify_locked(waiters: &mut WaitList, state: &AtomicUsize, curr: usize) -> Op
728
769
// concurrently change as holding the lock is required to
729
770
// transition **out** of `WAITING`.
730
771
//
731
- // Get a pending waiter
732
- let waiter = waiters. pop_back ( ) . unwrap ( ) ;
772
+ // Get a pending waiter using one of the available dequeue strategies.
773
+ let waiter = match strategy {
774
+ NotifyOneStrategy :: Fifo => waiters. pop_back ( ) . unwrap ( ) ,
775
+ NotifyOneStrategy :: Lifo => waiters. pop_front ( ) . unwrap ( ) ,
776
+ } ;
733
777
734
778
// Safety: we never make mutable references to waiters.
735
779
let waiter = unsafe { waiter. as_ref ( ) } ;
@@ -738,7 +782,9 @@ fn notify_locked(waiters: &mut WaitList, state: &AtomicUsize, curr: usize) -> Op
738
782
let waker = unsafe { waiter. waker . with_mut ( |waker| ( * waker) . take ( ) ) } ;
739
783
740
784
// This waiter is unlinked and will not be shared ever again, release it.
741
- waiter. notification . store_release ( Notification :: One ) ;
785
+ waiter
786
+ . notification
787
+ . store_release ( Notification :: One ( strategy) ) ;
742
788
743
789
if waiters. is_empty ( ) {
744
790
// As this the **final** waiter in the list, the state
@@ -1137,8 +1183,10 @@ impl Drop for Notified<'_> {
1137
1183
// See if the node was notified but not received. In this case, if
1138
1184
// the notification was triggered via `notify_one`, it must be sent
1139
1185
// to the next waiter.
1140
- if notification == Some ( Notification :: One ) {
1141
- if let Some ( waker) = notify_locked ( & mut waiters, & notify. state , notify_state) {
1186
+ if let Some ( Notification :: One ( strategy) ) = notification {
1187
+ if let Some ( waker) =
1188
+ notify_locked ( & mut waiters, & notify. state , notify_state, strategy)
1189
+ {
1142
1190
drop ( waiters) ;
1143
1191
waker. wake ( ) ;
1144
1192
}
0 commit comments