@ -541,6 +541,7 @@ bool DBImpl::SuperVersion::Unref() {
}
}
void DBImpl : : SuperVersion : : Cleanup ( ) {
void DBImpl : : SuperVersion : : Cleanup ( ) {
db - > mutex_ . AssertHeld ( ) ;
assert ( refs . load ( std : : memory_order_relaxed ) = = 0 ) ;
assert ( refs . load ( std : : memory_order_relaxed ) = = 0 ) ;
imm - > Unref ( & to_delete ) ;
imm - > Unref ( & to_delete ) ;
MemTable * m = mem - > Unref ( ) ;
MemTable * m = mem - > Unref ( ) ;
@ -552,6 +553,7 @@ void DBImpl::SuperVersion::Cleanup() {
void DBImpl : : SuperVersion : : Init ( MemTable * new_mem , MemTableListVersion * new_imm ,
void DBImpl : : SuperVersion : : Init ( MemTable * new_mem , MemTableListVersion * new_imm ,
Version * new_current ) {
Version * new_current ) {
db - > mutex_ . AssertHeld ( ) ;
mem = new_mem ;
mem = new_mem ;
imm = new_imm ;
imm = new_imm ;
current = new_current ;
current = new_current ;
@ -2960,8 +2962,10 @@ Status DBImpl::GetImpl(const ReadOptions& options,
// acquiring mutex for this operation, we use atomic Swap() on the thread
// acquiring mutex for this operation, we use atomic Swap() on the thread
// local pointer to guarantee exclusive access. If the thread local pointer
// local pointer to guarantee exclusive access. If the thread local pointer
// is being used while a new SuperVersion is installed, the cached
// is being used while a new SuperVersion is installed, the cached
// SuperVersion can become stale. It will eventually get refreshed either
// SuperVersion can become stale. In that case, the background thread would
// on the next GetImpl() call or next SuperVersion installation.
// have swapped in kSVObsolete. We re-check the value at the end of
// Get, with an atomic compare and swap. The superversion will be released
// if detected to be stale.
void * ptr = local_sv_ - > Swap ( SuperVersion : : kSVInUse ) ;
void * ptr = local_sv_ - > Swap ( SuperVersion : : kSVInUse ) ;
// Invariant:
// Invariant:
// (1) Scrape (always) installs kSVObsolete in ThreadLocal storage
// (1) Scrape (always) installs kSVObsolete in ThreadLocal storage
@ -2976,7 +2980,10 @@ Status DBImpl::GetImpl(const ReadOptions& options,
SuperVersion * sv_to_delete = nullptr ;
SuperVersion * sv_to_delete = nullptr ;
if ( sv & & sv - > Unref ( ) ) {
if ( sv & & sv - > Unref ( ) ) {
RecordTick ( options_ . statistics . get ( ) , NUMBER_SUPERVERSION_CLEANUPS ) ;
mutex_ . Lock ( ) ;
mutex_ . Lock ( ) ;
// TODO underlying resources held by superversion (sst files) might
// not be released until the next background job.
sv - > Cleanup ( ) ;
sv - > Cleanup ( ) ;
sv_to_delete = sv ;
sv_to_delete = sv ;
} else {
} else {
@ -3051,15 +3058,12 @@ Status DBImpl::GetImpl(const ReadOptions& options,
if ( unref_sv ) {
if ( unref_sv ) {
// Release SuperVersion
// Release SuperVersion
bool delete_sv = false ;
if ( sv - > Unref ( ) ) {
if ( sv - > Unref ( ) ) {
mutex_ . Lock ( ) ;
mutex_ . Lock ( ) ;
sv - > Cleanup ( ) ;
sv - > Cleanup ( ) ;
mutex_ . Unlock ( ) ;
mutex_ . Unlock ( ) ;
delete_sv = true ;
}
if ( delete_sv ) {
delete sv ;
delete sv ;
RecordTick ( options_ . statistics . get ( ) , NUMBER_SUPERVERSION_CLEANUPS ) ;
}
}
RecordTick ( options_ . statistics . get ( ) , NUMBER_SUPERVERSION_RELEASES ) ;
RecordTick ( options_ . statistics . get ( ) , NUMBER_SUPERVERSION_RELEASES ) ;
}
}