@ -84,6 +84,22 @@ struct DBImpl::CompactionState { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					struct  DBImpl : : DeletionState  {  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // the set of all live files that cannot be deleted
   
			
		
	
		
		
			
				
					
					  std : : set < uint64_t >  live ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // a list of all siles that exists in the db directory
   
			
		
	
		
		
			
				
					
					  std : : vector < std : : string >  allfiles ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // the current filenumber, lognumber and prevlognumber
   
			
		
	
		
		
			
				
					
					  // that corresponds to the set of files in 'live'.
   
			
		
	
		
		
			
				
					
					  uint64_t  filenumber ,  lognumber ,  prevlognumber ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // the list of all files to be evicted from the table cahce
   
			
		
	
		
		
			
				
					
					  std : : vector < uint64_t >  files_to_evict ;   
			
		
	
		
		
			
				
					
					} ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// Fix user-supplied options to be reasonable
 // Fix user-supplied options to be reasonable
  
			
		
	
		
		
			
				
					
					template  < class  T , class  V > template  < class  T , class  V >  
			
		
	
		
		
			
				
					
					static  void  ClipToRange ( T *  ptr ,  V  minvalue ,  V  maxvalue )  { static  void  ClipToRange ( T *  ptr ,  V  minvalue ,  V  maxvalue )  {  
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -248,7 +264,11 @@ void DBImpl::MaybeIgnoreError(Status* s) const { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  DBImpl : : DeleteObsoleteFiles ( )  { // Returns the list of live files in 'live' and the list
  
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					// of all files in the filesystem in 'allfiles'.
  
			
		
	
		
		
			
				
					
					void  DBImpl : : FindObsoleteFiles ( DeletionState &  deletion_state )  {  
			
		
	
		
		
			
				
					
					  mutex_ . AssertHeld ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // if deletion is disabled, do nothing
    // if deletion is disabled, do nothing
   
			
		
	
		
		
			
				
					
					  if  ( disable_delete_obsolete_files_ )  {    if  ( disable_delete_obsolete_files_ )  {   
			
		
	
		
		
			
				
					
					    return ;      return ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -267,39 +287,51 @@ void DBImpl::DeleteObsoleteFiles() { 
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Make a set of all of the live files
    // Make a set of all of the live files
   
			
		
	
		
		
			
				
					
					  std : : set < uint64_t >  live  =  pending_outputs_ ;    deletion_state . live  =  pending_outputs_ ;   
			
				
				
			
		
	
		
		
			
				
					
					  versions_ - > AddLiveFiles ( & live ) ;    versions_ - > AddLiveFiles ( & deletion_state . live ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  std : : vector < std : : string >  filenames ;    // set of all files in the directory
   
			
				
				
			
		
	
		
		
			
				
					
					  env_ - > GetChildren ( dbname_ ,  & filenames ) ;  // Ignoring errors on purpose
    env_ - > GetChildren ( dbname_ ,  & deletion_state . allfiles ) ;  // Ignore errors
   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // store the current filenum, lognum, etc
   
			
		
	
		
		
			
				
					
					  deletion_state . filenumber  =  versions_ - > ManifestFileNumber ( ) ;   
			
		
	
		
		
			
				
					
					  deletion_state . lognumber  =  versions_ - > LogNumber ( ) ;   
			
		
	
		
		
			
				
					
					  deletion_state . prevlognumber  =  versions_ - > PrevLogNumber ( ) ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					// Diffs the files listed in filenames and those that do not
  
			
		
	
		
		
			
				
					
					// belong to live files are posibly removed. If the removed file
  
			
		
	
		
		
			
				
					
					// is a sst file, then it returns the file number in files_to_evict.
  
			
		
	
		
		
			
				
					
					// It is not necesary to hold the mutex when invoking this method.
  
			
		
	
		
		
			
				
					
					void  DBImpl : : PurgeObsoleteFiles ( DeletionState &  state )  {  
			
		
	
		
		
			
				
					
					  uint64_t  number ;    uint64_t  number ;   
			
		
	
		
		
			
				
					
					  FileType  type ;    FileType  type ;   
			
		
	
		
		
			
				
					
					  std : : vector < std : : string >  old_log_files ;    std : : vector < std : : string >  old_log_files ;   
			
		
	
		
		
			
				
					
					  for  ( size_t  i  =  0 ;  i  <  filenames . size ( ) ;  i + + )  {    for  ( size_t  i  =  0 ;  i  <  state . all files. size ( ) ;  i + + )  {   
			
				
				
			
		
	
		
		
			
				
					
					    if  ( ParseFileName ( filenames [ i ] ,  & number ,  & type ) )  {      if  ( ParseFileName ( state . all files[ i ] ,  & number ,  & type ) )  {   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					      bool  keep  =  true ;        bool  keep  =  true ;   
			
		
	
		
		
			
				
					
					      switch  ( type )  {        switch  ( type )  {   
			
		
	
		
		
			
				
					
					        case  kLogFile :          case  kLogFile :   
			
		
	
		
		
			
				
					
					          keep  =  ( ( number  > =  versions_ - > LogNumber ( ) )  | |            keep  =  ( ( number  > =  state . lognumber )  | |   
			
				
				
			
		
	
		
		
			
				
					
					                  ( number  = =  versions_ - > PrevLogNumber ( ) ) ) ;                    ( number  = =  state . prevlognumber ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					          break ;            break ;   
			
		
	
		
		
			
				
					
					        case  kDescriptorFile :          case  kDescriptorFile :   
			
		
	
		
		
			
				
					
					          // Keep my manifest file, and any newer incarnations'
            // Keep my manifest file, and any newer incarnations'
   
			
		
	
		
		
			
				
					
					          // (in case there is a race that allows other incarnations)
            // (in case there is a race that allows other incarnations)
   
			
		
	
		
		
			
				
					
					          keep  =  ( number  > =  versions_ - > ManifestFileNumber ( ) ) ;            keep  =  ( number  > =  state . filenumber ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					          break ;            break ;   
			
		
	
		
		
			
				
					
					        case  kTableFile :          case  kTableFile :   
			
		
	
		
		
			
				
					
					          keep  =  ( live . find ( number )  ! =  live . end ( ) ) ;            keep  =  ( state . live . find ( number )  ! =  state . live . end ( ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					          break ;            break ;   
			
		
	
		
		
			
				
					
					        case  kTempFile :          case  kTempFile :   
			
		
	
		
		
			
				
					
					          // Any temp files that are currently being written to must
            // Any temp files that are currently being written to must
   
			
		
	
		
		
			
				
					
					          // be recorded in pending_outputs_, which is inserted into "live"
            // be recorded in pending_outputs_, which is inserted into "live"
   
			
		
	
		
		
			
				
					
					          keep  =  ( live . find ( number )  ! =  live . end ( ) ) ;            keep  =  ( state . live . find ( number )  ! =  state . live . end ( ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					          break ;            break ;   
			
		
	
		
		
			
				
					
					        case  kInfoLogFile :          case  kInfoLogFile :   
			
		
	
		
		
			
				
					
					          keep  =  true ;            keep  =  true ;   
			
		
	
		
		
			
				
					
					          if  ( number  ! =  0 )  {            if  ( number  ! =  0 )  {   
			
		
	
		
		
			
				
					
					            old_log_files . push_back ( filenam es [ i ] ) ;              old_log_files . push_back ( state . all files[ i ] ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					          }            }   
			
		
	
		
		
			
				
					
					          break ;            break ;   
			
		
	
		
		
			
				
					
					        case  kCurrentFile :          case  kCurrentFile :   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -310,12 +342,13 @@ void DBImpl::DeleteObsoleteFiles() { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					      if  ( ! keep )  {        if  ( ! keep )  {   
			
		
	
		
		
			
				
					
					        if  ( type  = =  kTableFile )  {          if  ( type  = =  kTableFile )  {   
			
		
	
		
		
			
				
					
					          table_cache_ - > Evict ( number ) ;            // record the files to be evicted from the cache
   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					          state . files_to_evict . push_back ( number ) ;   
			
		
	
		
		
			
				
					
					        }          }   
			
		
	
		
		
			
				
					
					        Log ( options_ . info_log ,  " Delete type=%d #%lld \n " ,          Log ( options_ . info_log ,  " Delete type=%d #%lld \n " ,   
			
		
	
		
		
			
				
					
					            int ( type ) ,              int ( type ) ,   
			
		
	
		
		
			
				
					
					            static_cast < unsigned  long  long > ( number ) ) ;              static_cast < unsigned  long  long > ( number ) ) ;   
			
		
	
		
		
			
				
					
					        Status  st  =  env_ - > DeleteFile ( dbname_  +  " / "  +  filenam es [ i ] ) ;          Status  st  =  env_ - > DeleteFile ( dbname_  +  " / "  +  state . all files[ i ] ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					        if ( ! st . ok ( ) )  {          if ( ! st . ok ( ) )  {   
			
		
	
		
		
			
				
					
					          Log ( options_ . info_log ,  " Delete type=%d #%lld FAILED \n " ,            Log ( options_ . info_log ,  " Delete type=%d #%lld FAILED \n " ,   
			
		
	
		
		
			
				
					
					              int ( type ) ,                int ( type ) ,   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -332,13 +365,32 @@ void DBImpl::DeleteObsoleteFiles() { 
			
		
	
		
		
			
				
					
					    std : : sort ( old_log_files . begin ( ) ,  old_log_files . end ( ) ) ;      std : : sort ( old_log_files . begin ( ) ,  old_log_files . end ( ) ) ;   
			
		
	
		
		
			
				
					
					    for  ( int  i  =  0 ;  i  > =  ( old_log_file_count  -  KEEP_LOG_FILE_NUM ) ;  i + + )  {      for  ( int  i  =  0 ;  i  > =  ( old_log_file_count  -  KEEP_LOG_FILE_NUM ) ;  i + + )  {   
			
		
	
		
		
			
				
					
					      std : : string &  to_delete  =  old_log_files . at ( i ) ;        std : : string &  to_delete  =  old_log_files . at ( i ) ;   
			
		
	
		
		
			
				
					
					      Log ( options_ . info_log ,  " Delete type=%d %s \n " ,        // Log(options_.info_log, "Delete type=%d %s\n",
   
			
				
				
			
		
	
		
		
			
				
					
					          int ( kInfoLogFile ) ,  to_delete . c_str ( ) ) ;         //     int(kInfoLogFile), to_delete.c_str());
   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					      env_ - > DeleteFile ( dbname_  +  " / "  +  to_delete ) ;        env_ - > DeleteFile ( dbname_  +  " / "  +  to_delete ) ;   
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  DBImpl : : EvictObsoleteFiles ( DeletionState &  state )  {  
			
		
	
		
		
			
				
					
					  mutex_ . AssertHeld ( ) ;   
			
		
	
		
		
			
				
					
					  for  ( unsigned  int  i  =  0 ;  i  <  state . files_to_evict . size ( ) ;  i + + )  {   
			
		
	
		
		
			
				
					
					    table_cache_ - > Evict ( state . files_to_evict [ i ] ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  DBImpl : : DeleteObsoleteFiles ( )  {  
			
		
	
		
		
			
				
					
					  mutex_ . AssertHeld ( ) ;   
			
		
	
		
		
			
				
					
					  DeletionState  deletion_state ;   
			
		
	
		
		
			
				
					
					  std : : set < uint64_t >  live ;   
			
		
	
		
		
			
				
					
					  std : : vector < std : : string >  allfiles ;   
			
		
	
		
		
			
				
					
					  std : : vector < uint64_t >  files_to_evict ;   
			
		
	
		
		
			
				
					
					  uint64_t  filenumber ,  lognumber ,  prevlognumber ;   
			
		
	
		
		
			
				
					
					  FindObsoleteFiles ( deletion_state ) ;   
			
		
	
		
		
			
				
					
					  PurgeObsoleteFiles ( deletion_state ) ;   
			
		
	
		
		
			
				
					
					  EvictObsoleteFiles ( deletion_state ) ;   
			
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					Status  DBImpl : : Recover ( VersionEdit *  edit )  { Status  DBImpl : : Recover ( VersionEdit *  edit )  {  
			
		
	
		
		
			
				
					
					  mutex_ . AssertHeld ( ) ;    mutex_ . AssertHeld ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -723,10 +775,11 @@ void DBImpl::BGWork(void* db) { 
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					void  DBImpl : : BackgroundCall ( )  { void  DBImpl : : BackgroundCall ( )  {  
			
		
	
		
		
			
				
					
					  DeletionState  deletion_state ;   
			
		
	
		
		
			
				
					
					  MutexLock  l ( & mutex_ ) ;    MutexLock  l ( & mutex_ ) ;   
			
		
	
		
		
			
				
					
					  assert ( bg_compaction_scheduled_ ) ;    assert ( bg_compaction_scheduled_ ) ;   
			
		
	
		
		
			
				
					
					  if  ( ! shutting_down_ . Acquire_Load ( ) )  {    if  ( ! shutting_down_ . Acquire_Load ( ) )  {   
			
		
	
		
		
			
				
					
					    Status  s  =  BackgroundCompaction ( ) ;      Status  s  =  BackgroundCompaction ( deletion_state ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					    if  ( ! s . ok ( ) )  {      if  ( ! s . ok ( ) )  {   
			
		
	
		
		
			
				
					
					      // Wait a little bit before retrying background compaction in
        // Wait a little bit before retrying background compaction in
   
			
		
	
		
		
			
				
					
					      // case this is an environmental problem and we do not want to
        // case this is an environmental problem and we do not want to
   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -741,6 +794,14 @@ void DBImpl::BackgroundCall() { 
			
		
	
		
		
			
				
					
					    }      }   
			
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // delete unnecessary files if any, this is done outside the mutex
   
			
		
	
		
		
			
				
					
					  if  ( ! deletion_state . live . empty ( ) )  {   
			
		
	
		
		
			
				
					
					    mutex_ . Unlock ( ) ;   
			
		
	
		
		
			
				
					
					    PurgeObsoleteFiles ( deletion_state ) ;   
			
		
	
		
		
			
				
					
					    mutex_ . Lock ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					  EvictObsoleteFiles ( deletion_state ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  bg_compaction_scheduled_  =  false ;    bg_compaction_scheduled_  =  false ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  MaybeScheduleLogDBDeployStats ( ) ;    MaybeScheduleLogDBDeployStats ( ) ;   
			
		
	
	
		
		
			
				
					
						
						
						
							
								 
						
					 
					@ -751,7 +812,7 @@ void DBImpl::BackgroundCall() { 
			
		
	
		
		
			
				
					
					  bg_cv_ . SignalAll ( ) ;    bg_cv_ . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					Status  DBImpl : : BackgroundCompaction ( )  { Status  DBImpl : : BackgroundCompaction ( DeletionState &  deletion_state )  {  
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  mutex_ . AssertHeld ( ) ;    mutex_ . AssertHeld ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  if  ( imm_  ! =  NULL )  {    if  ( imm_  ! =  NULL )  {   
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -801,7 +862,7 @@ Status DBImpl::BackgroundCompaction() { 
			
		
	
		
		
			
				
					
					    status  =  DoCompactionWork ( compact ) ;      status  =  DoCompactionWork ( compact ) ;   
			
		
	
		
		
			
				
					
					    CleanupCompaction ( compact ) ;      CleanupCompaction ( compact ) ;   
			
		
	
		
		
			
				
					
					    c - > ReleaseInputs ( ) ;      c - > ReleaseInputs ( ) ;   
			
		
	
		
		
			
				
					
					    DeleteObsoleteFiles ( ) ;      FindObsoleteFiles ( deletion_state ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  delete  c ;    delete  c ;