@ -1264,7 +1264,14 @@ Status DBImpl::FlushMemTableToOutputFile(ColumnFamilyData* cfd,
Status DBImpl : : CompactRange ( const ColumnFamilyHandle & column_family ,
Status DBImpl : : CompactRange ( const ColumnFamilyHandle & column_family ,
const Slice * begin , const Slice * end ,
const Slice * begin , const Slice * end ,
bool reduce_level , int target_level ) {
bool reduce_level , int target_level ) {
Status s = FlushMemTable ( default_cfd_ , FlushOptions ( ) ) ;
mutex_ . Lock ( ) ;
auto cfd = versions_ - > GetColumnFamilySet ( ) - > GetColumnFamily ( column_family . id ) ;
mutex_ . Unlock ( ) ;
// this is asserting because client calling DB methods with undefined
// ColumnFamilyHandle is undefined behavior.
assert ( cfd ! = nullptr ) ;
Status s = FlushMemTable ( cfd , FlushOptions ( ) ) ;
if ( ! s . ok ( ) ) {
if ( ! s . ok ( ) ) {
LogFlush ( options_ . info_log ) ;
LogFlush ( options_ . info_log ) ;
return s ;
return s ;
@ -1273,8 +1280,8 @@ Status DBImpl::CompactRange(const ColumnFamilyHandle& column_family,
int max_level_with_files = 1 ;
int max_level_with_files = 1 ;
{
{
MutexLock l ( & mutex_ ) ;
MutexLock l ( & mutex_ ) ;
Version * base = default_ cfd_ - > current ( ) ;
Version * base = cfd - > current ( ) ;
for ( int level = 1 ; level < NumberLevels ( ) ; level + + ) {
for ( int level = 1 ; level < cfd - > NumberLevels ( ) ; level + + ) {
if ( base - > OverlapInLevel ( level , begin , end ) ) {
if ( base - > OverlapInLevel ( level , begin , end ) ) {
max_level_with_files = level ;
max_level_with_files = level ;
}
}
@ -1285,9 +1292,9 @@ Status DBImpl::CompactRange(const ColumnFamilyHandle& column_family,
// bottom-most level, the output level will be the same as input one
// bottom-most level, the output level will be the same as input one
if ( options_ . compaction_style = = kCompactionStyleUniversal | |
if ( options_ . compaction_style = = kCompactionStyleUniversal | |
level = = max_level_with_files ) {
level = = max_level_with_files ) {
s = RunManualCompaction ( level , level , begin , end ) ;
s = RunManualCompaction ( cfd , level , level , begin , end ) ;
} else {
} else {
s = RunManualCompaction ( level , level + 1 , begin , end ) ;
s = RunManualCompaction ( cfd , level , level + 1 , begin , end ) ;
}
}
if ( ! s . ok ( ) ) {
if ( ! s . ok ( ) ) {
LogFlush ( options_ . info_log ) ;
LogFlush ( options_ . info_log ) ;
@ -1296,7 +1303,7 @@ Status DBImpl::CompactRange(const ColumnFamilyHandle& column_family,
}
}
if ( reduce_level ) {
if ( reduce_level ) {
s = ReFitLevel ( max_level_with_files , target_level ) ;
s = ReFitLevel ( cfd , max_level_with_files , target_level ) ;
}
}
LogFlush ( options_ . info_log ) ;
LogFlush ( options_ . info_log ) ;
@ -1304,15 +1311,15 @@ Status DBImpl::CompactRange(const ColumnFamilyHandle& column_family,
}
}
// return the same level if it cannot be moved
// return the same level if it cannot be moved
int DBImpl : : FindMinimumEmptyLevelFitting ( int level ) {
int DBImpl : : FindMinimumEmptyLevelFitting ( ColumnFamilyData * cfd , int level ) {
mutex_ . AssertHeld ( ) ;
mutex_ . AssertHeld ( ) ;
Version * current = default_ cfd_ - > current ( ) ;
Version * current = cfd - > current ( ) ;
int minimum_level = level ;
int minimum_level = level ;
for ( int i = level - 1 ; i > 0 ; - - i ) {
for ( int i = level - 1 ; i > 0 ; - - i ) {
// stop if level i is not empty
// stop if level i is not empty
if ( current - > NumLevelFiles ( i ) > 0 ) break ;
if ( current - > NumLevelFiles ( i ) > 0 ) break ;
// stop if level i is too small (cannot fit the level files)
// stop if level i is too small (cannot fit the level files)
if ( default_ cfd_ - > compaction_picker ( ) - > MaxBytesForLevel ( i ) <
if ( cfd - > compaction_picker ( ) - > MaxBytesForLevel ( i ) <
current - > NumLevelBytes ( level ) ) {
current - > NumLevelBytes ( level ) ) {
break ;
break ;
}
}
@ -1322,8 +1329,8 @@ int DBImpl::FindMinimumEmptyLevelFitting(int level) {
return minimum_level ;
return minimum_level ;
}
}
Status DBImpl : : ReFitLevel ( int level , int target_level ) {
Status DBImpl : : ReFitLevel ( ColumnFamilyData * cfd , int level , int target_level ) {
assert ( level < NumberLevels ( ) ) ;
assert ( level < cfd - > NumberLevels ( ) ) ;
SuperVersion * superversion_to_free = nullptr ;
SuperVersion * superversion_to_free = nullptr ;
SuperVersion * new_superversion = new SuperVersion ( ) ;
SuperVersion * new_superversion = new SuperVersion ( ) ;
@ -1351,7 +1358,7 @@ Status DBImpl::ReFitLevel(int level, int target_level) {
// move to a smaller level
// move to a smaller level
int to_level = target_level ;
int to_level = target_level ;
if ( target_level < 0 ) {
if ( target_level < 0 ) {
to_level = FindMinimumEmptyLevelFitting ( level ) ;
to_level = FindMinimumEmptyLevelFitting ( cfd , level ) ;
}
}
assert ( to_level < = level ) ;
assert ( to_level < = level ) ;
@ -1359,10 +1366,11 @@ Status DBImpl::ReFitLevel(int level, int target_level) {
Status status ;
Status status ;
if ( to_level < level ) {
if ( to_level < level ) {
Log ( options_ . info_log , " Before refitting: \n %s " ,
Log ( options_ . info_log , " Before refitting: \n %s " ,
default_ cfd_ - > current ( ) - > DebugString ( ) . data ( ) ) ;
cfd - > current ( ) - > DebugString ( ) . data ( ) ) ;
VersionEdit edit ;
VersionEdit edit ;
for ( const auto & f : default_cfd_ - > current ( ) - > files_ [ level ] ) {
edit . SetColumnFamily ( cfd - > GetID ( ) ) ;
for ( const auto & f : cfd - > current ( ) - > files_ [ level ] ) {
edit . DeleteFile ( level , f - > number ) ;
edit . DeleteFile ( level , f - > number ) ;
edit . AddFile ( to_level , f - > number , f - > file_size , f - > smallest , f - > largest ,
edit . AddFile ( to_level , f - > number , f - > file_size , f - > smallest , f - > largest ,
f - > smallest_seqno , f - > largest_seqno ) ;
f - > smallest_seqno , f - > largest_seqno ) ;
@ -1370,16 +1378,15 @@ Status DBImpl::ReFitLevel(int level, int target_level) {
Log ( options_ . info_log , " Apply version edit: \n %s " ,
Log ( options_ . info_log , " Apply version edit: \n %s " ,
edit . DebugString ( ) . data ( ) ) ;
edit . DebugString ( ) . data ( ) ) ;
status = versions_ - > LogAndApply ( default_cfd_ , & edit , & mutex_ ,
status = versions_ - > LogAndApply ( cfd , & edit , & mutex_ , db_directory_ . get ( ) ) ;
db_directory_ . get ( ) ) ;
superversion_to_free = cfd - > InstallSuperVersion ( new_superversion ) ;
superversion_to_free = default_cfd_ - > InstallSuperVersion ( new_superversion ) ;
new_superversion = nullptr ;
new_superversion = nullptr ;
Log ( options_ . info_log , " LogAndApply: %s \n " , status . ToString ( ) . data ( ) ) ;
Log ( options_ . info_log , " LogAndApply: %s \n " , status . ToString ( ) . data ( ) ) ;
if ( status . ok ( ) ) {
if ( status . ok ( ) ) {
Log ( options_ . info_log , " After refitting: \n %s " ,
Log ( options_ . info_log , " After refitting: \n %s " ,
default_ cfd_ - > current ( ) - > DebugString ( ) . data ( ) ) ;
cfd - > current ( ) - > DebugString ( ) . data ( ) ) ;
}
}
}
}
@ -1607,15 +1614,15 @@ Status DBImpl::AppendSortedWalsOfType(const std::string& path,
return status ;
return status ;
}
}
Status DBImpl : : RunManualCompaction ( int input_level ,
Status DBImpl : : RunManualCompaction ( ColumnFamilyData * cfd , int input_level ,
int output_level ,
int output_level , const Slice * begin ,
const Slice * begin ,
const Slice * end ) {
const Slice * end ) {
assert ( input_level > = 0 ) ;
assert ( input_level > = 0 ) ;
InternalKey begin_storage , end_storage ;
InternalKey begin_storage , end_storage ;
ManualCompaction manual ;
ManualCompaction manual ;
manual . cfd = cfd ;
manual . input_level = input_level ;
manual . input_level = input_level ;
manual . output_level = output_level ;
manual . output_level = output_level ;
manual . done = false ;
manual . done = false ;
@ -1630,7 +1637,7 @@ Status DBImpl::RunManualCompaction(int input_level,
manual . begin = & begin_storage ;
manual . begin = & begin_storage ;
}
}
if ( end = = nullptr | |
if ( end = = nullptr | |
options_ . compaction_style = = kCompactionStyleUniversal ) {
cfd - > options ( ) - > compaction_style = = kCompactionStyleUniversal ) {
manual . end = nullptr ;
manual . end = nullptr ;
} else {
} else {
end_storage = InternalKey ( * end , 0 , static_cast < ValueType > ( 0 ) ) ;
end_storage = InternalKey ( * end , 0 , static_cast < ValueType > ( 0 ) ) ;
@ -1686,7 +1693,7 @@ Status DBImpl::TEST_CompactRange(int level,
int output_level = ( options_ . compaction_style = = kCompactionStyleUniversal )
int output_level = ( options_ . compaction_style = = kCompactionStyleUniversal )
? level
? level
: level + 1 ;
: level + 1 ;
return RunManualCompaction ( level , output_level , begin , end ) ;
return RunManualCompaction ( default_cfd_ , level , output_level , begin , end ) ;
}
}
Status DBImpl : : FlushMemTable ( ColumnFamilyData * cfd ,
Status DBImpl : : FlushMemTable ( ColumnFamilyData * cfd ,
@ -1946,8 +1953,8 @@ Status DBImpl::BackgroundCompaction(bool* madeProgress,
if ( is_manual ) {
if ( is_manual ) {
ManualCompaction * m = manual_compaction_ ;
ManualCompaction * m = manual_compaction_ ;
assert ( m - > in_progress ) ;
assert ( m - > in_progress ) ;
c . reset ( default_cfd_ - > CompactRange ( m - > input_level , m - > output_level ,
c . reset ( m - > cfd - > CompactRange ( m - > input_level , m - > output_level , m - > begin ,
m - > begin , m - > end , & manual_end ) ) ;
m - > end , & manual_end ) ) ;
if ( ! c ) {
if ( ! c ) {
m - > done = true ;
m - > done = true ;
}
}
@ -1962,7 +1969,12 @@ Status DBImpl::BackgroundCompaction(bool* madeProgress,
? " (end) "
? " (end) "
: manual_end - > DebugString ( ) . c_str ( ) ) ) ;
: manual_end - > DebugString ( ) . c_str ( ) ) ) ;
} else if ( ! options_ . disable_auto_compactions ) {
} else if ( ! options_ . disable_auto_compactions ) {
c . reset ( default_cfd_ - > PickCompaction ( ) ) ;
for ( auto cfd : * versions_ - > GetColumnFamilySet ( ) ) {
c . reset ( cfd - > PickCompaction ( ) ) ;
if ( c ! = nullptr ) {
break ;
}
}
}
}
Status status ;
Status status ;
@ -1977,23 +1989,23 @@ Status DBImpl::BackgroundCompaction(bool* madeProgress,
c - > edit ( ) - > AddFile ( c - > level ( ) + 1 , f - > number , f - > file_size ,
c - > edit ( ) - > AddFile ( c - > level ( ) + 1 , f - > number , f - > file_size ,
f - > smallest , f - > largest ,
f - > smallest , f - > largest ,
f - > smallest_seqno , f - > largest_seqno ) ;
f - > smallest_seqno , f - > largest_seqno ) ;
status = versions_ - > LogAndApply ( default_cfd_ , c - > edit ( ) , & mutex_ ,
status = versions_ - > LogAndApply ( c - > column_family_data ( ) , c - > edit ( ) , & mutex_ ,
db_directory_ . get ( ) ) ;
db_directory_ . get ( ) ) ;
InstallSuperVersion ( default_cfd_ , deletion_state ) ;
InstallSuperVersion ( c - > column_family_data ( ) , deletion_state ) ;
Version : : LevelSummaryStorage tmp ;
Version : : LevelSummaryStorage tmp ;
Log ( options_ . info_log , " Moved #%lld to level-%d %lld bytes %s: %s \n " ,
Log ( options_ . info_log , " Moved #%lld to level-%d %lld bytes %s: %s \n " ,
static_cast < unsigned long long > ( f - > number ) , c - > level ( ) + 1 ,
static_cast < unsigned long long > ( f - > number ) , c - > level ( ) + 1 ,
static_cast < unsigned long long > ( f - > file_size ) ,
static_cast < unsigned long long > ( f - > file_size ) ,
status . ToString ( ) . c_str ( ) , default_ cfd_ - > current ( ) - > LevelSummary ( & tmp ) ) ;
status . ToString ( ) . c_str ( ) , c - > input_version ( ) - > LevelSummary ( & tmp ) ) ;
default_ cfd_ - > compaction_picker ( ) - > ReleaseCompactionFiles ( c . get ( ) , status ) ;
c - > ReleaseCompactionFiles ( status ) ;
* madeProgress = true ;
* madeProgress = true ;
} else {
} else {
MaybeScheduleFlushOrCompaction ( ) ; // do more compaction work in parallel.
MaybeScheduleFlushOrCompaction ( ) ; // do more compaction work in parallel.
CompactionState * compact = new CompactionState ( c . get ( ) ) ;
CompactionState * compact = new CompactionState ( c . get ( ) ) ;
status = DoCompactionWork ( compact , deletion_state ) ;
status = DoCompactionWork ( compact , deletion_state ) ;
CleanupCompaction ( compact , status ) ;
CleanupCompaction ( compact , status ) ;
default_ cfd_ - > compaction_picker ( ) - > ReleaseCompactionFiles ( c . get ( ) , status ) ;
c - > ReleaseCompactionFiles ( status ) ;
c - > ReleaseInputs ( ) ;
c - > ReleaseInputs ( ) ;
* madeProgress = true ;
* madeProgress = true ;
}
}
@ -2123,8 +2135,9 @@ Status DBImpl::OpenCompactionOutputFile(CompactionState* compact) {
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
// Over-estimate slightly so we don't end up just barely crossing
// Over-estimate slightly so we don't end up just barely crossing
// the threshold.
// the threshold.
ColumnFamilyData * cfd = compact - > compaction - > column_family_data ( ) ;
compact - > outfile - > SetPreallocationBlockSize (
compact - > outfile - > SetPreallocationBlockSize (
1.1 * default_ cfd_ - > compaction_picker ( ) - > MaxFileSizeForLevel (
1.1 * cfd - > compaction_picker ( ) - > MaxFileSizeForLevel (
compact - > compaction - > output_level ( ) ) ) ;
compact - > compaction - > output_level ( ) ) ) ;
CompressionType compression_type = GetCompressionType (
CompressionType compression_type = GetCompressionType (
@ -2228,8 +2241,9 @@ Status DBImpl::InstallCompactionResults(CompactionState* compact) {
compact - > compaction - > output_level ( ) , out . number , out . file_size ,
compact - > compaction - > output_level ( ) , out . number , out . file_size ,
out . smallest , out . largest , out . smallest_seqno , out . largest_seqno ) ;
out . smallest , out . largest , out . smallest_seqno , out . largest_seqno ) ;
}
}
return versions_ - > LogAndApply ( default_cfd_ , compact - > compaction - > edit ( ) ,
return versions_ - > LogAndApply ( compact - > compaction - > column_family_data ( ) ,
& mutex_ , db_directory_ . get ( ) ) ;
compact - > compaction - > edit ( ) , & mutex_ ,
db_directory_ . get ( ) ) ;
}
}
//
//
@ -2264,13 +2278,12 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
DeletionState & deletion_state ) {
DeletionState & deletion_state ) {
assert ( compact ) ;
assert ( compact ) ;
int64_t imm_micros = 0 ; // Micros spent doing imm_ compactions
int64_t imm_micros = 0 ; // Micros spent doing imm_ compactions
ColumnFamilyData * cfd = compact - > compaction - > column_family_data ( ) ;
Log ( options_ . info_log ,
Log ( options_ . info_log ,
" Compacting %d@%d + %d@%d files, score %.2f slots available %d " ,
" [CF %u] Compacting %d@%d + %d@%d files, score %.2f slots available %d " ,
compact - > compaction - > num_input_files ( 0 ) ,
cfd - > GetID ( ) , compact - > compaction - > num_input_files ( 0 ) ,
compact - > compaction - > level ( ) ,
compact - > compaction - > level ( ) , compact - > compaction - > num_input_files ( 1 ) ,
compact - > compaction - > num_input_files ( 1 ) ,
compact - > compaction - > output_level ( ) , compact - > compaction - > score ( ) ,
compact - > compaction - > output_level ( ) ,
compact - > compaction - > score ( ) ,
options_ . max_background_compactions - bg_compaction_scheduled_ ) ;
options_ . max_background_compactions - bg_compaction_scheduled_ ) ;
char scratch [ 256 ] ;
char scratch [ 256 ] ;
compact - > compaction - > Summary ( scratch , sizeof ( scratch ) ) ;
compact - > compaction - > Summary ( scratch , sizeof ( scratch ) ) ;
@ -2321,7 +2334,7 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
MergeHelper merge ( user_comparator ( ) , options_ . merge_operator . get ( ) ,
MergeHelper merge ( user_comparator ( ) , options_ . merge_operator . get ( ) ,
options_ . info_log . get ( ) ,
options_ . info_log . get ( ) ,
false /* internal key corruption is expected */ ) ;
false /* internal key corruption is expected */ ) ;
auto compaction_filter = options_ . compaction_filter ;
auto compaction_filter = cfd - > options ( ) - > compaction_filter ;
std : : unique_ptr < CompactionFilter > compaction_filter_from_factory = nullptr ;
std : : unique_ptr < CompactionFilter > compaction_filter_from_factory = nullptr ;
if ( ! compaction_filter ) {
if ( ! compaction_filter ) {
auto context = compact - > GetFilterContext ( ) ;
auto context = compact - > GetFilterContext ( ) ;
@ -2333,12 +2346,12 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
for ( ; input - > Valid ( ) & & ! shutting_down_ . Acquire_Load ( ) ; ) {
for ( ; input - > Valid ( ) & & ! shutting_down_ . Acquire_Load ( ) ; ) {
// Prioritize immutable compaction work
// Prioritize immutable compaction work
// TODO: remove memtable flush from normal compaction work
// TODO: remove memtable flush from normal compaction work
if ( default_ cfd_ - > imm ( ) - > imm_flush_needed . NoBarrier_Load ( ) ! = nullptr ) {
if ( cfd - > imm ( ) - > imm_flush_needed . NoBarrier_Load ( ) ! = nullptr ) {
const uint64_t imm_start = env_ - > NowMicros ( ) ;
const uint64_t imm_start = env_ - > NowMicros ( ) ;
LogFlush ( options_ . info_log ) ;
LogFlush ( options_ . info_log ) ;
mutex_ . Lock ( ) ;
mutex_ . Lock ( ) ;
if ( default_ cfd_ - > imm ( ) - > IsFlushPending ( ) ) {
if ( cfd - > imm ( ) - > IsFlushPending ( ) ) {
FlushMemTableToOutputFile ( default_ cfd_ , nullptr , deletion_state ) ;
FlushMemTableToOutputFile ( cfd , nullptr , deletion_state ) ;
bg_cv_ . SignalAll ( ) ; // Wakeup MakeRoomForWrite() if necessary
bg_cv_ . SignalAll ( ) ; // Wakeup MakeRoomForWrite() if necessary
}
}
mutex_ . Unlock ( ) ;
mutex_ . Unlock ( ) ;
@ -2388,11 +2401,9 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
// the entry with a delete marker.
// the entry with a delete marker.
bool value_changed = false ;
bool value_changed = false ;
compaction_filter_value . clear ( ) ;
compaction_filter_value . clear ( ) ;
bool to_delete =
bool to_delete = compaction_filter - > Filter (
compaction_filter - > Filter ( compact - > compaction - > level ( ) ,
compact - > compaction - > level ( ) , ikey . user_key , value ,
ikey . user_key , value ,
& compaction_filter_value , & value_changed ) ;
& compaction_filter_value ,
& value_changed ) ;
if ( to_delete ) {
if ( to_delete ) {
// make a copy of the original key
// make a copy of the original key
delete_key . assign ( key . data ( ) , key . data ( ) + key . size ( ) ) ;
delete_key . assign ( key . data ( ) , key . data ( ) + key . size ( ) ) ;
@ -2410,7 +2421,6 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
value = compaction_filter_value ;
value = compaction_filter_value ;
}
}
}
}
}
}
// If there are no snapshots, then this kv affect visibility at tip.
// If there are no snapshots, then this kv affect visibility at tip.
@ -2649,7 +2659,7 @@ Status DBImpl::DoCompactionWork(CompactionState* compact,
if ( status . ok ( ) ) {
if ( status . ok ( ) ) {
status = InstallCompactionResults ( compact ) ;
status = InstallCompactionResults ( compact ) ;
InstallSuperVersion ( default_ cfd_ , deletion_state ) ;
InstallSuperVersion ( cfd , deletion_state ) ;
}
}
Version : : LevelSummaryStorage tmp ;
Version : : LevelSummaryStorage tmp ;
Log ( options_ . info_log ,
Log ( options_ . info_log ,
@ -2847,7 +2857,7 @@ Status DBImpl::GetImpl(const ReadOptions& options,
mutex_ . Lock ( ) ;
mutex_ . Lock ( ) ;
auto cfd = versions_ - > GetColumnFamilySet ( ) - > GetColumnFamily ( column_family . id ) ;
auto cfd = versions_ - > GetColumnFamilySet ( ) - > GetColumnFamily ( column_family . id ) ;
// this is asserting because client calling Get() with undefined
// this is asserting because client calling DB methods with undefined
// ColumnFamilyHandle is undefined behavior.
// ColumnFamilyHandle is undefined behavior.
assert ( cfd ! = nullptr ) ;
assert ( cfd ! = nullptr ) ;
SuperVersion * get_version = cfd - > GetSuperVersion ( ) - > Ref ( ) ;
SuperVersion * get_version = cfd - > GetSuperVersion ( ) - > Ref ( ) ;
@ -3538,7 +3548,8 @@ bool DBImpl::GetProperty(const ColumnFamilyHandle& column_family,
const Slice & property , std : : string * value ) {
const Slice & property , std : : string * value ) {
value - > clear ( ) ;
value - > clear ( ) ;
MutexLock l ( & mutex_ ) ;
MutexLock l ( & mutex_ ) ;
return internal_stats_ . GetProperty ( property , value , default_cfd_ ) ;
auto cfd = versions_ - > GetColumnFamilySet ( ) - > GetColumnFamily ( column_family . id ) ;
return internal_stats_ . GetProperty ( property , value , cfd ) ;
}
}
void DBImpl : : GetApproximateSizes ( const ColumnFamilyHandle & column_family ,
void DBImpl : : GetApproximateSizes ( const ColumnFamilyHandle & column_family ,