@@ -88,18 +88,33 @@ impl From<ThreadId> for u64 {
88
88
}
89
89
}
90
90
91
+ /// Keeps track of what the thread is blocked on.
92
+ #[ derive( Debug , Copy , Clone , PartialEq , Eq ) ]
93
+ pub enum BlockReason {
94
+ /// The thread tried to join the specified thread and is blocked until that
95
+ /// thread terminates.
96
+ Join ( ThreadId ) ,
97
+ /// Waiting for time to pass.
98
+ Sleep ,
99
+ /// Blocked on a mutex.
100
+ Mutex ( MutexId ) ,
101
+ /// Blocked on a condition variable.
102
+ Condvar ( CondvarId ) ,
103
+ /// Blocked on a reader-writer lock.
104
+ RwLock ( RwLockId ) ,
105
+ /// Blocled on a Futex variable.
106
+ Futex { addr : u64 } ,
107
+ /// Blocked on an InitOnce.
108
+ InitOnce ( InitOnceId ) ,
109
+ }
110
+
91
111
/// The state of a thread.
92
112
#[ derive( Debug , Copy , Clone , PartialEq , Eq ) ]
93
113
pub enum ThreadState {
94
114
/// The thread is enabled and can be executed.
95
115
Enabled ,
96
- /// The thread tried to join the specified thread and is blocked until that
97
- /// thread terminates.
98
- BlockedOnJoin ( ThreadId ) ,
99
- /// The thread is blocked on some synchronization primitive. It is the
100
- /// responsibility of the synchronization primitives to track threads that
101
- /// are blocked by them.
102
- BlockedOnSync ,
116
+ /// The thread is blocked on something.
117
+ Blocked ( BlockReason ) ,
103
118
/// The thread has terminated its execution. We do not delete terminated
104
119
/// threads (FIXME: why?).
105
120
Terminated ,
@@ -296,17 +311,17 @@ impl VisitProvenance for Frame<'_, '_, Provenance, FrameExtra<'_>> {
296
311
297
312
/// A specific moment in time.
298
313
#[ derive( Debug ) ]
299
- pub enum Time {
314
+ pub enum CallbackTime {
300
315
Monotonic ( Instant ) ,
301
316
RealTime ( SystemTime ) ,
302
317
}
303
318
304
- impl Time {
319
+ impl CallbackTime {
305
320
/// How long do we have to wait from now until the specified time?
306
321
fn get_wait_time ( & self , clock : & Clock ) -> Duration {
307
322
match self {
308
- Time :: Monotonic ( instant) => instant. duration_since ( clock. now ( ) ) ,
309
- Time :: RealTime ( time) =>
323
+ CallbackTime :: Monotonic ( instant) => instant. duration_since ( clock. now ( ) ) ,
324
+ CallbackTime :: RealTime ( time) =>
310
325
time. duration_since ( SystemTime :: now ( ) ) . unwrap_or ( Duration :: new ( 0 , 0 ) ) ,
311
326
}
312
327
}
@@ -318,7 +333,7 @@ impl Time {
318
333
/// conditional variable, the signal handler deletes the callback.
319
334
struct TimeoutCallbackInfo < ' mir , ' tcx > {
320
335
/// The callback should be called no earlier than this time.
321
- call_time : Time ,
336
+ call_time : CallbackTime ,
322
337
/// The called function.
323
338
callback : TimeoutCallback < ' mir , ' tcx > ,
324
339
}
@@ -539,7 +554,8 @@ impl<'mir, 'tcx: 'mir> ThreadManager<'mir, 'tcx> {
539
554
self . threads [ joined_thread_id] . join_status = ThreadJoinStatus :: Joined ;
540
555
if self . threads [ joined_thread_id] . state != ThreadState :: Terminated {
541
556
// The joined thread is still running, we need to wait for it.
542
- self . active_thread_mut ( ) . state = ThreadState :: BlockedOnJoin ( joined_thread_id) ;
557
+ self . active_thread_mut ( ) . state =
558
+ ThreadState :: Blocked ( BlockReason :: Join ( joined_thread_id) ) ;
543
559
trace ! (
544
560
"{:?} blocked on {:?} when trying to join" ,
545
561
self . active_thread,
@@ -569,10 +585,11 @@ impl<'mir, 'tcx: 'mir> ThreadManager<'mir, 'tcx> {
569
585
throw_ub_format ! ( "trying to join itself" ) ;
570
586
}
571
587
588
+ // Sanity check `join_status`.
572
589
assert ! (
573
- self . threads
574
- . iter ( )
575
- . all ( |thread| thread . state != ThreadState :: BlockedOnJoin ( joined_thread_id ) ) ,
590
+ self . threads. iter ( ) . all ( |thread| {
591
+ thread . state != ThreadState :: Blocked ( BlockReason :: Join ( joined_thread_id ) )
592
+ } ) ,
576
593
"this thread already has threads waiting for its termination"
577
594
) ;
578
595
@@ -594,16 +611,17 @@ impl<'mir, 'tcx: 'mir> ThreadManager<'mir, 'tcx> {
594
611
}
595
612
596
613
/// Put the thread into the blocked state.
597
- fn block_thread ( & mut self , thread : ThreadId ) {
614
+ fn block_thread ( & mut self , thread : ThreadId , reason : BlockReason ) {
598
615
let state = & mut self . threads [ thread] . state ;
599
616
assert_eq ! ( * state, ThreadState :: Enabled ) ;
600
- * state = ThreadState :: BlockedOnSync ;
617
+ * state = ThreadState :: Blocked ( reason ) ;
601
618
}
602
619
603
620
/// Put the blocked thread into the enabled state.
604
- fn unblock_thread ( & mut self , thread : ThreadId ) {
621
+ /// Sanity-checks that the thread previously was blocked for the right reason.
622
+ fn unblock_thread ( & mut self , thread : ThreadId , reason : BlockReason ) {
605
623
let state = & mut self . threads [ thread] . state ;
606
- assert_eq ! ( * state, ThreadState :: BlockedOnSync ) ;
624
+ assert_eq ! ( * state, ThreadState :: Blocked ( reason ) ) ;
607
625
* state = ThreadState :: Enabled ;
608
626
}
609
627
@@ -622,7 +640,7 @@ impl<'mir, 'tcx: 'mir> ThreadManager<'mir, 'tcx> {
622
640
fn register_timeout_callback (
623
641
& mut self ,
624
642
thread : ThreadId ,
625
- call_time : Time ,
643
+ call_time : CallbackTime ,
626
644
callback : TimeoutCallback < ' mir , ' tcx > ,
627
645
) {
628
646
self . timeout_callbacks
@@ -683,7 +701,7 @@ impl<'mir, 'tcx: 'mir> ThreadManager<'mir, 'tcx> {
683
701
// Check if we need to unblock any threads.
684
702
let mut joined_threads = vec ! [ ] ; // store which threads joined, we'll need it
685
703
for ( i, thread) in self . threads . iter_enumerated_mut ( ) {
686
- if thread. state == ThreadState :: BlockedOnJoin ( self . active_thread ) {
704
+ if thread. state == ThreadState :: Blocked ( BlockReason :: Join ( self . active_thread ) ) {
687
705
// The thread has terminated, mark happens-before edge to joining thread
688
706
if data_race. is_some ( ) {
689
707
joined_threads. push ( i) ;
@@ -999,13 +1017,13 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
999
1017
}
1000
1018
1001
1019
#[ inline]
1002
- fn block_thread ( & mut self , thread : ThreadId ) {
1003
- self . eval_context_mut ( ) . machine . threads . block_thread ( thread) ;
1020
+ fn block_thread ( & mut self , thread : ThreadId , reason : BlockReason ) {
1021
+ self . eval_context_mut ( ) . machine . threads . block_thread ( thread, reason ) ;
1004
1022
}
1005
1023
1006
1024
#[ inline]
1007
- fn unblock_thread ( & mut self , thread : ThreadId ) {
1008
- self . eval_context_mut ( ) . machine . threads . unblock_thread ( thread) ;
1025
+ fn unblock_thread ( & mut self , thread : ThreadId , reason : BlockReason ) {
1026
+ self . eval_context_mut ( ) . machine . threads . unblock_thread ( thread, reason ) ;
1009
1027
}
1010
1028
1011
1029
#[ inline]
@@ -1027,11 +1045,11 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
1027
1045
fn register_timeout_callback (
1028
1046
& mut self ,
1029
1047
thread : ThreadId ,
1030
- call_time : Time ,
1048
+ call_time : CallbackTime ,
1031
1049
callback : TimeoutCallback < ' mir , ' tcx > ,
1032
1050
) {
1033
1051
let this = self . eval_context_mut ( ) ;
1034
- if !this. machine . communicate ( ) && matches ! ( call_time, Time :: RealTime ( ..) ) {
1052
+ if !this. machine . communicate ( ) && matches ! ( call_time, CallbackTime :: RealTime ( ..) ) {
1035
1053
panic ! ( "cannot have `RealTime` callback with isolation enabled!" )
1036
1054
}
1037
1055
this. machine . threads . register_timeout_callback ( thread, call_time, callback) ;
0 commit comments