@@ -4,7 +4,7 @@ use smallvec::SmallVec;
4
4
use std:: {
5
5
cmp:: Ordering ,
6
6
fmt:: Debug ,
7
- ops:: { Index , IndexMut , Shr } ,
7
+ ops:: { Index , Shr } ,
8
8
} ;
9
9
10
10
use super :: data_race:: NaReadType ;
@@ -92,7 +92,7 @@ impl VTimestamp {
92
92
}
93
93
94
94
#[ inline]
95
- pub fn set_read_type ( & mut self , read_type : NaReadType ) {
95
+ pub ( super ) fn set_read_type ( & mut self , read_type : NaReadType ) {
96
96
self . time_and_read_type = Self :: encode_time_and_read_type ( self . time ( ) , read_type) ;
97
97
}
98
98
@@ -138,7 +138,7 @@ pub struct VClock(SmallVec<[VTimestamp; SMALL_VECTOR]>);
138
138
impl VClock {
139
139
/// Create a new vector-clock containing all zeros except
140
140
/// for a value at the given index
141
- pub fn new_with_index ( index : VectorIdx , timestamp : VTimestamp ) -> VClock {
141
+ pub ( super ) fn new_with_index ( index : VectorIdx , timestamp : VTimestamp ) -> VClock {
142
142
let len = index. index ( ) + 1 ;
143
143
let mut vec = smallvec:: smallvec![ VTimestamp :: ZERO ; len] ;
144
144
vec[ index. index ( ) ] = timestamp;
@@ -147,10 +147,16 @@ impl VClock {
147
147
148
148
/// Load the internal timestamp slice in the vector clock
149
149
#[ inline]
150
- pub fn as_slice ( & self ) -> & [ VTimestamp ] {
150
+ pub ( super ) fn as_slice ( & self ) -> & [ VTimestamp ] {
151
+ debug_assert ! ( !self . 0 . last( ) . is_some_and( |t| t. time( ) == 0 ) ) ;
151
152
self . 0 . as_slice ( )
152
153
}
153
154
155
+ #[ inline]
156
+ pub ( super ) fn index_mut ( & mut self , index : VectorIdx ) -> & mut VTimestamp {
157
+ self . 0 . as_mut_slice ( ) . get_mut ( index. to_u32 ( ) as usize ) . unwrap ( )
158
+ }
159
+
154
160
/// Get a mutable slice to the internal vector with minimum `min_len`
155
161
/// elements. To preserve invariants, the caller must modify
156
162
/// the `min_len`-1 nth element to a non-zero value
@@ -166,7 +172,7 @@ impl VClock {
166
172
/// Increment the vector clock at a known index
167
173
/// this will panic if the vector index overflows
168
174
#[ inline]
169
- pub fn increment_index ( & mut self , idx : VectorIdx , current_span : Span ) {
175
+ pub ( super ) fn increment_index ( & mut self , idx : VectorIdx , current_span : Span ) {
170
176
let idx = idx. index ( ) ;
171
177
let mut_slice = self . get_mut_with_min_len ( idx + 1 ) ;
172
178
let idx_ref = & mut mut_slice[ idx] ;
@@ -190,28 +196,36 @@ impl VClock {
190
196
}
191
197
}
192
198
193
- /// Set the element at the current index of the vector
194
- pub fn set_at_index ( & mut self , other : & Self , idx : VectorIdx ) {
199
+ /// Set the element at the current index of the vector. May only increase elements.
200
+ pub ( super ) fn set_at_index ( & mut self , other : & Self , idx : VectorIdx ) {
201
+ let new_timestamp = other[ idx] ;
202
+ // Setting to 0 is different, since the last element cannot be 0.
203
+ if new_timestamp. time ( ) == 0 {
204
+ if idx. index ( ) >= self . 0 . len ( ) {
205
+ // This index does not even exist yet in our clock. Just do nothing.
206
+ return ;
207
+ }
208
+ // This changes an existing element. Since it can only increase, that
209
+ // can never make the last element 0.
210
+ }
211
+
195
212
let mut_slice = self . get_mut_with_min_len ( idx. index ( ) + 1 ) ;
213
+ let mut_timestamp = & mut mut_slice[ idx. index ( ) ] ;
196
214
197
- let prev_span = mut_slice [ idx . index ( ) ] . span ;
215
+ let prev_span = mut_timestamp . span ;
198
216
199
- mut_slice[ idx. index ( ) ] = other[ idx] ;
217
+ assert ! ( * mut_timestamp <= new_timestamp, "set_at_index: may only increase the timestamp" ) ;
218
+ * mut_timestamp = new_timestamp;
200
219
201
- let span = & mut mut_slice [ idx . index ( ) ] . span ;
220
+ let span = & mut mut_timestamp . span ;
202
221
* span = span. substitute_dummy ( prev_span) ;
203
222
}
204
223
205
224
/// Set the vector to the all-zero vector
206
225
#[ inline]
207
- pub fn set_zero_vector ( & mut self ) {
226
+ pub ( super ) fn set_zero_vector ( & mut self ) {
208
227
self . 0 . clear ( ) ;
209
228
}
210
-
211
- /// Return if this vector is the all-zero vector
212
- pub fn is_zero_vector ( & self ) -> bool {
213
- self . 0 . is_empty ( )
214
- }
215
229
}
216
230
217
231
impl Clone for VClock {
@@ -407,13 +421,6 @@ impl Index<VectorIdx> for VClock {
407
421
}
408
422
}
409
423
410
- impl IndexMut < VectorIdx > for VClock {
411
- #[ inline]
412
- fn index_mut ( & mut self , index : VectorIdx ) -> & mut VTimestamp {
413
- self . 0 . as_mut_slice ( ) . get_mut ( index. to_u32 ( ) as usize ) . unwrap ( )
414
- }
415
- }
416
-
417
424
/// Test vector clock ordering operations
418
425
/// data-race detection is tested in the external
419
426
/// test suite
@@ -553,4 +560,15 @@ mod tests {
553
560
"Invalid alt (>=):\n l: {l:?}\n r: {r:?}"
554
561
) ;
555
562
}
563
+
564
+ #[ test]
565
+ fn set_index_to_0 ( ) {
566
+ let mut clock1 = from_slice ( & [ 0 , 1 , 2 , 3 ] ) ;
567
+ let clock2 = from_slice ( & [ 0 , 2 , 3 , 4 , 0 , 5 ] ) ;
568
+ // Naively, this would extend clock1 with a new index and set it to 0, making
569
+ // the last index 0. Make sure that does not happen.
570
+ clock1. set_at_index ( & clock2, VectorIdx ( 4 ) ) ;
571
+ // This must not have made the last element 0.
572
+ assert ! ( clock1. 0 . last( ) . unwrap( ) . time( ) != 0 ) ;
573
+ }
556
574
}
0 commit comments