@ -14,6 +14,8 @@
namespace rocksdb {
namespace rocksdb {
// The structure that manages compaction input files associated
// with the same physical level.
struct CompactionInputFiles {
struct CompactionInputFiles {
int level ;
int level ;
std : : vector < FileMetaData * > files ;
std : : vector < FileMetaData * > files ;
@ -36,35 +38,63 @@ class Compaction {
~ Compaction ( ) ;
~ Compaction ( ) ;
// Returns the level associated to the specified compaction input level.
// Returns the level associated to the specified compaction input level.
// If input_level is not specified, then input_level is set to 0.
// If compaction_input_level is not specified, then input_level is set to 0.
int level ( int input_level = 0 ) const { return inputs_ [ input_level ] . level ; }
int level ( int compaction_input_level = 0 ) const {
return inputs_ [ compaction_input_level ] . level ;
}
// Outputs will go to this level
// Outputs will go to this level
int output_level ( ) const { return out_level_ ; }
int output_level ( ) const { return output_level_ ; }
// Returns the number of input levels in this compaction.
int num_input_levels ( ) const { return inputs_ . size ( ) ; }
// Return the object that holds the edits to the descriptor done
// Return the object that holds the edits to the descriptor done
// by this compaction.
// by this compaction.
VersionEdit * edit ( ) { return edit_ ; }
VersionEdit * edit ( ) const { return edit_ ; }
// "which" must be either 0 or 1
// Returns the number of input files associated to the specified
int num_input_files ( int which ) const { return inputs_ [ which ] . size ( ) ; }
// compaction input level.
// The function will return 0 if when "compaction_input_level" < 0
// or "compaction_input_level" >= "num_input_levels()".
int num_input_files ( int compaction_input_level ) const {
if ( compaction_input_level > = 0 & &
compaction_input_level < inputs_ . size ( ) ) {
return inputs_ [ compaction_input_level ] . size ( ) ;
}
return 0 ;
}
// Returns input version of the compaction
// Returns input version of the compaction
Version * input_version ( ) const { return input_version_ ; }
Version * input_version ( ) const { return input_version_ ; }
// Returns the ColumnFamilyData associated with the compaction.
ColumnFamilyData * column_family_data ( ) const { return cfd_ ; }
ColumnFamilyData * column_family_data ( ) const { return cfd_ ; }
// Return the ith input file at "level()+which" ("which" must be 0 or 1).
// Returns the file meta data of the 'i'th input file at the
FileMetaData * input ( int which , int i ) const { return inputs_ [ which ] [ i ] ; }
// specified compaction input level.
// REQUIREMENT: "compaction_input_level" must be >= 0 and
// < "input_levels()"
FileMetaData * input ( int compaction_input_level , int i ) const {
assert ( compaction_input_level < inputs_ . size ( ) & &
compaction_input_level > = 0 ) ;
return inputs_ [ compaction_input_level ] [ i ] ;
}
// Returns the list of FileMataData associated with the specified
// Returns the list of file meta data of the specified compaction
// compaction input level.
// input level.
std : : vector < FileMetaData * > * inputs ( int which ) {
// REQUIREMENT: "compaction_input_level" must be >= 0 and
return & inputs_ [ which ] . files ;
// < "input_levels()"
std : : vector < FileMetaData * > * const inputs ( int compaction_input_level ) {
assert ( compaction_input_level < inputs_ . size ( ) & &
compaction_input_level > = 0 ) ;
return & inputs_ [ compaction_input_level ] . files ;
}
}
// Return the input_level file
// Returns the FileLevel of the specified compaction input level.
FileLevel * input_levels ( int which ) { return & input_levels_ [ which ] ; }
FileLevel * input_levels ( int compaction_input_level ) {
return & input_levels_ [ compaction_input_level ] ;
}
// Maximum size of files to build during this compaction.
// Maximum size of files to build during this compaction.
uint64_t MaxOutputFileSize ( ) const { return max_output_file_size_ ; }
uint64_t MaxOutputFileSize ( ) const { return max_output_file_size_ ; }
@ -83,16 +113,17 @@ class Compaction {
// moving a single input file to the next level (no merging or splitting)
// moving a single input file to the next level (no merging or splitting)
bool IsTrivialMove ( ) const ;
bool IsTrivialMove ( ) const ;
// If true, just delete all files in inputs_[0]
// If true, then the comaction can be done by simply deleting input files.
bool IsDeletionCompaction ( ) const ;
bool IsDeletionCompaction ( ) const {
return deletion_compaction_ ;
}
// Add all inputs to this compaction as delete operations to *edit.
// Add all inputs to this compaction as delete operations to *edit.
void AddInputDeletions ( VersionEdit * edit ) ;
void AddInputDeletions ( VersionEdit * edit ) ;
// Returns true if the information we have available guarantees that
// Returns true if the available information we have guarantees that
// the compaction is producing data in "level+1" for which no data exists
// the input "user_key" does not exist in any level beyond "output_level()".
// in levels greater than "level+1".
bool KeyNotExistsBeyondOutputLevel ( const Slice & user_key ) ;
bool IsBaseLevelForKey ( const Slice & user_key ) ;
// Returns true iff we should stop building the current output
// Returns true iff we should stop building the current output
// before processing "internal_key".
// before processing "internal_key".
@ -106,6 +137,9 @@ class Compaction {
// Delete this compaction from the list of running compactions.
// Delete this compaction from the list of running compactions.
void ReleaseCompactionFiles ( Status status ) ;
void ReleaseCompactionFiles ( Status status ) ;
// Returns the summary of the compaction in "output" with maximum "len"
// in bytes. The caller is responsible for the memory management of
// "output".
void Summary ( char * output , int len ) ;
void Summary ( char * output , int len ) ;
// Return the score that was used to pick this compaction run.
// Return the score that was used to pick this compaction run.
@ -120,9 +154,9 @@ class Compaction {
// Was this compaction triggered manually by the client?
// Was this compaction triggered manually by the client?
bool IsManualCompaction ( ) { return is_manual_compaction_ ; }
bool IsManualCompaction ( ) { return is_manual_compaction_ ; }
// Returns a number of byte that the output file should be preallocated to
// Returns the size in bytes that the output file should be preallocated to.
// In level compaction, that is max_file_size_. In universal compaction, that
// In level compaction, that is max_file_size_. In universal compaction, that
// is the sum of all input file sizes
// is the sum of all input file sizes.
uint64_t OutputFilePreallocationSize ( ) ;
uint64_t OutputFilePreallocationSize ( ) ;
private :
private :
@ -131,13 +165,13 @@ class Compaction {
friend class FIFOCompactionPicker ;
friend class FIFOCompactionPicker ;
friend class LevelCompactionPicker ;
friend class LevelCompactionPicker ;
Compaction ( Version * input_version , int level , int out_level ,
Compaction ( Version * input_version , int start_ level, int out_level ,
uint64_t target_file_size , uint64_t max_grandparent_overlap_bytes ,
uint64_t target_file_size , uint64_t max_grandparent_overlap_bytes ,
uint32_t output_path_id , CompressionType output_compression ,
uint32_t output_path_id , CompressionType output_compression ,
bool seek_compaction = false , bool deletion_compaction = false ) ;
bool seek_compaction = false , bool deletion_compaction = false ) ;
int level_ ;
const int start_ level_; // the lowest level to be compacted
int out_level_ ; // levels to which output files are stored
const int outp ut_level_ ; // levels to which output files are stored
uint64_t max_output_file_size_ ;
uint64_t max_output_file_size_ ;
uint64_t max_grandparent_overlap_bytes_ ;
uint64_t max_grandparent_overlap_bytes_ ;
Version * input_version_ ;
Version * input_version_ ;
@ -149,24 +183,26 @@ class Compaction {
uint32_t output_path_id_ ;
uint32_t output_path_id_ ;
CompressionType output_compression_ ;
CompressionType output_compression_ ;
bool seek_compaction_ ;
bool seek_compaction_ ;
// if true, just delete files in inputs_[0]
// If true, then the comaction can be done by simply deleting input files.
bool deletion_compaction_ ;
bool deletion_compaction_ ;
// Each compaction reads inputs from "level_" and "level_+1"
// Compaction input files organized by level.
CompactionInputFiles inputs_ [ 2 ] ; // The two sets of inputs
autovector < CompactionInputFiles > inputs_ ;
// A copy of inputs_, organized more closely in memory
// A copy of inputs_, organized more closely in memory
autovector < FileLevel , 2 > input_levels_ ;
autovector < FileLevel , 2 > input_levels_ ;
// State used to check for number of of overlapping grandparent files
// State used to check for number of of overlapping grandparent files
// (parent == level_ + 1, grandparent == level_ + 2)
// (grandparent == "output_level_ + 1")
// This vector is updated by Version::GetOverlappingInputs().
std : : vector < FileMetaData * > grandparents_ ;
std : : vector < FileMetaData * > grandparents_ ;
size_t grandparent_index_ ; // Index in grandparent_starts_
size_t grandparent_index_ ; // Index in grandparent_starts_
bool seen_key_ ; // Some output key has been seen
bool seen_key_ ; // Some output key has been seen
uint64_t overlapped_bytes_ ; // Bytes of overlap between current output
uint64_t overlapped_bytes_ ; // Bytes of overlap between current output
// and grandparent files
// and grandparent files
int base_index_ ; // index of the file in files_[level_]
int base_index_ ; // index of the file in files_[start_level_]
int parent_index_ ; // index of some file with same range in files_[level_+1]
int parent_index_ ; // index of some file with same range in
// files_[start_level_+1]
double score_ ; // score that was used to pick this compaction.
double score_ ; // score that was used to pick this compaction.
// Is this compaction creating a file in the bottom most level?
// Is this compaction creating a file in the bottom most level?
@ -177,17 +213,21 @@ class Compaction {
// Is this compaction requested by the client?
// Is this compaction requested by the client?
bool is_manual_compaction_ ;
bool is_manual_compaction_ ;
// level_ptrs_ holds indices into input_version_->levels_: our state
// "level_ptrs_" holds indices into "input_version_->levels_", where each
// is that we are positioned at one of the file ranges for each
// index remembers which file of an associated level we are currently used
// higher level than the ones involved in this compaction (i.e. for
// to check KeyNotExistsBeyondOutputLevel() for deletion operation.
// all L >= level_ + 2).
// As it is for checking KeyNotExistsBeyondOutputLevel(), it only
// records indices for all levels beyond "output_level_".
std : : vector < size_t > level_ptrs_ ;
std : : vector < size_t > level_ptrs_ ;
// mark (or clear) all files that are being compacted
// mark (or clear) all files that are being compacted
void MarkFilesBeingCompacted ( bool ) ;
void MarkFilesBeingCompacted ( bool mark_as_compacted ) ;
// Initialize whether compaction producing files at the bottommost level
// Initialize whether the compaction is producing files at the
void SetupBottomMostLevel ( bool isManual ) ;
// bottommost level.
//
// @see BottomMostLevel()
void SetupBottomMostLevel ( bool is_manual ) ;
// In case of compaction error, reset the nextIndex that is used
// In case of compaction error, reset the nextIndex that is used
// to pick up the next file to be compacted from files_by_size_
// to pick up the next file to be compacted from files_by_size_