@@ -547,9 +547,9 @@ impl MemoryCellClocks {
547
547
) -> Result < ( ) , DataRace > {
548
548
trace ! ( "Unsynchronized read with vectors: {:#?} :: {:#?}" , self , thread_clocks) ;
549
549
if !current_span. is_dummy ( ) {
550
- thread_clocks. clock [ index] . span = current_span;
550
+ thread_clocks. clock . index_mut ( index) . span = current_span;
551
551
}
552
- thread_clocks. clock [ index] . set_read_type ( read_type) ;
552
+ thread_clocks. clock . index_mut ( index) . set_read_type ( read_type) ;
553
553
if self . write_was_before ( & thread_clocks. clock ) {
554
554
let race_free = if let Some ( atomic) = self . atomic ( ) {
555
555
// We must be ordered-after all atomic accesses, reads and writes.
@@ -577,7 +577,7 @@ impl MemoryCellClocks {
577
577
) -> Result < ( ) , DataRace > {
578
578
trace ! ( "Unsynchronized write with vectors: {:#?} :: {:#?}" , self , thread_clocks) ;
579
579
if !current_span. is_dummy ( ) {
580
- thread_clocks. clock [ index] . span = current_span;
580
+ thread_clocks. clock . index_mut ( index) . span = current_span;
581
581
}
582
582
if self . write_was_before ( & thread_clocks. clock ) && self . read <= thread_clocks. clock {
583
583
let race_free = if let Some ( atomic) = self . atomic ( ) {
@@ -1701,49 +1701,34 @@ impl GlobalState {
1701
1701
format ! ( "thread `{thread_name}`" )
1702
1702
}
1703
1703
1704
- /// Acquire a lock, express that the previous call of
1705
- /// `validate_lock_release` must happen before this .
1704
+ /// Acquire the given clock into the given thread, establishing synchronization with
1705
+ /// the moment when that clock snapshot was taken via `release_clock` .
1706
1706
/// As this is an acquire operation, the thread timestamp is not
1707
1707
/// incremented.
1708
- pub fn validate_lock_acquire ( & self , lock : & VClock , thread : ThreadId ) {
1709
- let ( _, mut clocks) = self . load_thread_state_mut ( thread) ;
1708
+ pub fn acquire_clock ( & self , lock : & VClock , thread : ThreadId ) {
1709
+ let ( _, mut clocks) = self . thread_state_mut ( thread) ;
1710
1710
clocks. clock . join ( lock) ;
1711
1711
}
1712
1712
1713
- /// Release a lock handle, express that this happens-before
1714
- /// any subsequent calls to `validate_lock_acquire`.
1715
- /// For normal locks this should be equivalent to `validate_lock_release_shared`
1716
- /// since an acquire operation should have occurred before, however
1717
- /// for futex & condvar operations this is not the case and this
1718
- /// operation must be used.
1719
- pub fn validate_lock_release ( & self , lock : & mut VClock , thread : ThreadId , current_span : Span ) {
1720
- let ( index, mut clocks) = self . load_thread_state_mut ( thread) ;
1721
- lock. clone_from ( & clocks. clock ) ;
1722
- clocks. increment_clock ( index, current_span) ;
1723
- }
1724
-
1725
- /// Release a lock handle, express that this happens-before
1726
- /// any subsequent calls to `validate_lock_acquire` as well
1727
- /// as any previous calls to this function after any
1728
- /// `validate_lock_release` calls.
1729
- /// For normal locks this should be equivalent to `validate_lock_release`.
1730
- /// This function only exists for joining over the set of concurrent readers
1731
- /// in a read-write lock and should not be used for anything else.
1732
- pub fn validate_lock_release_shared (
1733
- & self ,
1734
- lock : & mut VClock ,
1735
- thread : ThreadId ,
1736
- current_span : Span ,
1737
- ) {
1738
- let ( index, mut clocks) = self . load_thread_state_mut ( thread) ;
1739
- lock. join ( & clocks. clock ) ;
1713
+ /// Returns the `release` clock of the given thread.
1714
+ /// Other threads can acquire this clock in the future to establish synchronization
1715
+ /// with this program point.
1716
+ pub fn release_clock ( & self , thread : ThreadId , current_span : Span ) -> Ref < ' _ , VClock > {
1717
+ // We increment the clock each time this happens, to ensure no two releases
1718
+ // can be confused with each other.
1719
+ let ( index, mut clocks) = self . thread_state_mut ( thread) ;
1740
1720
clocks. increment_clock ( index, current_span) ;
1721
+ drop ( clocks) ;
1722
+ // To return a read-only view, we need to release the RefCell
1723
+ // and borrow it again.
1724
+ let ( _index, clocks) = self . thread_state ( thread) ;
1725
+ Ref :: map ( clocks, |c| & c. clock )
1741
1726
}
1742
1727
1743
1728
/// Load the vector index used by the given thread as well as the set of vector clocks
1744
1729
/// used by the thread.
1745
1730
#[ inline]
1746
- fn load_thread_state_mut ( & self , thread : ThreadId ) -> ( VectorIdx , RefMut < ' _ , ThreadClockSet > ) {
1731
+ fn thread_state_mut ( & self , thread : ThreadId ) -> ( VectorIdx , RefMut < ' _ , ThreadClockSet > ) {
1747
1732
let index = self . thread_info . borrow ( ) [ thread]
1748
1733
. vector_index
1749
1734
. expect ( "Loading thread state for thread with no assigned vector" ) ;
@@ -1752,17 +1737,26 @@ impl GlobalState {
1752
1737
( index, clocks)
1753
1738
}
1754
1739
1740
+ /// Load the vector index used by the given thread as well as the set of vector clocks
1741
+ /// used by the thread.
1742
+ #[ inline]
1743
+ fn thread_state ( & self , thread : ThreadId ) -> ( VectorIdx , Ref < ' _ , ThreadClockSet > ) {
1744
+ let index = self . thread_info . borrow ( ) [ thread]
1745
+ . vector_index
1746
+ . expect ( "Loading thread state for thread with no assigned vector" ) ;
1747
+ let ref_vector = self . vector_clocks . borrow ( ) ;
1748
+ let clocks = Ref :: map ( ref_vector, |vec| & vec[ index] ) ;
1749
+ ( index, clocks)
1750
+ }
1751
+
1755
1752
/// Load the current vector clock in use and the current set of thread clocks
1756
1753
/// in use for the vector.
1757
1754
#[ inline]
1758
1755
pub ( super ) fn current_thread_state (
1759
1756
& self ,
1760
1757
thread_mgr : & ThreadManager < ' _ , ' _ > ,
1761
1758
) -> ( VectorIdx , Ref < ' _ , ThreadClockSet > ) {
1762
- let index = self . current_index ( thread_mgr) ;
1763
- let ref_vector = self . vector_clocks . borrow ( ) ;
1764
- let clocks = Ref :: map ( ref_vector, |vec| & vec[ index] ) ;
1765
- ( index, clocks)
1759
+ self . thread_state ( thread_mgr. get_active_thread_id ( ) )
1766
1760
}
1767
1761
1768
1762
/// Load the current vector clock in use and the current set of thread clocks
@@ -1772,10 +1766,7 @@ impl GlobalState {
1772
1766
& self ,
1773
1767
thread_mgr : & ThreadManager < ' _ , ' _ > ,
1774
1768
) -> ( VectorIdx , RefMut < ' _ , ThreadClockSet > ) {
1775
- let index = self . current_index ( thread_mgr) ;
1776
- let ref_vector = self . vector_clocks . borrow_mut ( ) ;
1777
- let clocks = RefMut :: map ( ref_vector, |vec| & mut vec[ index] ) ;
1778
- ( index, clocks)
1769
+ self . thread_state_mut ( thread_mgr. get_active_thread_id ( ) )
1779
1770
}
1780
1771
1781
1772
/// Return the current thread, should be the same
0 commit comments