@ -16,50 +16,72 @@ class SnapshotList;
/ / Each SnapshotImpl corresponds to a particular sequence number .
/ / Each SnapshotImpl corresponds to a particular sequence number .
class SnapshotImpl : public Snapshot {
class SnapshotImpl : public Snapshot {
public :
public :
SequenceNumber number_ ; / / const after creation
SnapshotImpl ( SequenceNumber sequence_number )
: sequence_number_ ( sequence_number ) { }
SequenceNumber sequence_number ( ) const { return sequence_number_ ; }
private :
private :
friend class SnapshotList ;
friend class SnapshotList ;
/ / SnapshotImpl is kept in a doubly - linked circular list
/ / SnapshotImpl is kept in a doubly - linked circular list . The SnapshotList
/ / implementation operates on the next / previous fields direcly .
SnapshotImpl * prev_ ;
SnapshotImpl * prev_ ;
SnapshotImpl * next_ ;
SnapshotImpl * next_ ;
SnapshotList * list_ ; / / just for sanity checks
const SequenceNumber sequence_number_ ;
# if !defined(NDEBUG)
SnapshotList * list_ = nullptr ;
# endif / / !defined(NDEBUG)
} ;
} ;
class SnapshotList {
class SnapshotList {
public :
public :
SnapshotList ( ) {
list _. prev_ = & list _;
list _. next_ = & list _;
SnapshotList ( ) : head_ ( 0 ) {
head _. prev_ = & head _;
head _. next_ = & head _;
}
}
bool empty ( ) const { return list_ . next_ = = & list_ ; }
SnapshotImpl * oldest ( ) const { assert ( ! empty ( ) ) ; return list_ . next_ ; }
SnapshotImpl * newest ( ) const { assert ( ! empty ( ) ) ; return list_ . prev_ ; }
const SnapshotImpl * New ( SequenceNumber seq ) {
SnapshotImpl * s = new SnapshotImpl ;
s - > number_ = seq ;
s - > list_ = this ;
s - > next_ = & list_ ;
s - > prev_ = list_ . prev_ ;
s - > prev_ - > next_ = s ;
s - > next_ - > prev_ = s ;
return s ;
bool empty ( ) const { return head_ . next_ = = & head_ ; }
SnapshotImpl * oldest ( ) const { assert ( ! empty ( ) ) ; return head_ . next_ ; }
SnapshotImpl * newest ( ) const { assert ( ! empty ( ) ) ; return head_ . prev_ ; }
/ / Creates a SnapshotImpl and appends it to the end of the list .
SnapshotImpl * New ( SequenceNumber sequence_number ) {
assert ( empty ( ) | | newest ( ) - > sequence_number_ < = sequence_number ) ;
SnapshotImpl * snapshot = new SnapshotImpl ( sequence_number ) ;
# if !defined(NDEBUG)
snapshot - > list_ = this ;
# endif / / !defined(NDEBUG)
snapshot - > next_ = & head_ ;
snapshot - > prev_ = head_ . prev_ ;
snapshot - > prev_ - > next_ = snapshot ;
snapshot - > next_ - > prev_ = snapshot ;
return snapshot ;
}
}
void Delete ( const SnapshotImpl * s ) {
assert ( s - > list_ = = this ) ;
s - > prev_ - > next_ = s - > next_ ;
s - > next_ - > prev_ = s - > prev_ ;
delete s ;
/ / Removes a SnapshotImpl from this list .
/ /
/ / The snapshot must have been created by calling New ( ) on this list .
/ /
/ / The snapshot pointer should not be const , because its memory is
/ / deallocated . However , that would force us to change DB : : ReleaseSnapshot ( ) ,
/ / which is in the API , and currently takes a const Snapshot .
void Delete ( const SnapshotImpl * snapshot ) {
# if !defined(NDEBUG)
assert ( snapshot - > list_ = = this ) ;
# endif / / !defined(NDEBUG)
snapshot - > prev_ - > next_ = snapshot - > next_ ;
snapshot - > next_ - > prev_ = snapshot - > prev_ ;
delete snapshot ;
}
}
private :
private :
/ / Dummy head of doubly - linked list of snapshots
/ / Dummy head of doubly - linked list of snapshots
SnapshotImpl list_ ;
SnapshotImpl head _;
} ;
} ;
} / / namespace leveldb
} / / namespace leveldb