@ -20,10 +20,11 @@
# include "rocksdb/rate_limiter.h"
# include "rocksdb/rate_limiter.h"
# include "rocksdb/transaction_log.h"
# include "rocksdb/transaction_log.h"
# include "port/port.h"
# include "port/port.h"
# include "util/sync_point.h"
# ifndef __STDC_FORMAT_MACROS
# ifndef __STDC_FORMAT_MACROS
# define __STDC_FORMAT_MACROS
# define __STDC_FORMAT_MACROS
# endif
# endif // __STDC_FORMAT_MACROS
# include <inttypes.h>
# include <inttypes.h>
# include <stdlib.h>
# include <stdlib.h>
@ -39,8 +40,6 @@
# include <thread>
# include <thread>
# include <unordered_map>
# include <unordered_map>
# include <unordered_set>
# include <unordered_set>
# include "port/port.h"
namespace rocksdb {
namespace rocksdb {
@ -269,43 +268,59 @@ class BackupEngineImpl : public BackupEngine {
}
}
Status PutLatestBackupFileContents ( uint32_t latest_backup ) ;
Status PutLatestBackupFileContents ( uint32_t latest_backup ) ;
// if size_limit == 0, there is no size limit, copy everything
Status CopyFile ( const std : : string & src , const std : : string & dst , Env * src_env ,
// If size_limit == 0, there is no size limit, copy everything.
Env * dst_env , bool sync , RateLimiter * rate_limiter ,
//
uint64_t * size = nullptr , uint32_t * checksum_value = nullptr ,
// Exactly one of src and contents must be non-empty.
uint64_t size_limit = 0 ,
//
std : : function < void ( ) > progress_callback = [ ] ( ) { } ) ;
// @param src If non-empty, the file is copied from this pathname.
// @param contents If non-empty, the file will be created with these contents.
Status CopyOrCreateFile ( const std : : string & src , const std : : string & dst ,
const std : : string & contents , Env * src_env ,
Env * dst_env , bool sync , RateLimiter * rate_limiter ,
uint64_t * size = nullptr ,
uint32_t * checksum_value = nullptr ,
uint64_t size_limit = 0 ,
std : : function < void ( ) > progress_callback = [ ] ( ) { } ) ;
Status CalculateChecksum ( const std : : string & src ,
Status CalculateChecksum ( const std : : string & src ,
Env * src_env ,
Env * src_env ,
uint64_t size_limit ,
uint64_t size_limit ,
uint32_t * checksum_value ) ;
uint32_t * checksum_value ) ;
struct CopyResult {
struct CopyOrCreate Result {
uint64_t size ;
uint64_t size ;
uint32_t checksum_value ;
uint32_t checksum_value ;
Status status ;
Status status ;
} ;
} ;
struct CopyWorkItem {
// Exactly one of src_path and contents must be non-empty. If src_path is
// non-empty, the file is copied from this pathname. Otherwise, if contents is
// non-empty, the file will be created at dst_path with these contents.
struct CopyOrCreateWorkItem {
std : : string src_path ;
std : : string src_path ;
std : : string dst_path ;
std : : string dst_path ;
std : : string contents ;
Env * src_env ;
Env * src_env ;
Env * dst_env ;
Env * dst_env ;
bool sync ;
bool sync ;
RateLimiter * rate_limiter ;
RateLimiter * rate_limiter ;
uint64_t size_limit ;
uint64_t size_limit ;
std : : promise < CopyResult > result ;
std : : promise < CopyOrCreate Result > result ;
std : : function < void ( ) > progress_callback ;
std : : function < void ( ) > progress_callback ;
CopyWorkItem ( ) { }
CopyOrCreate WorkItem ( ) { }
CopyWorkItem ( const CopyWorkItem & ) = delete ;
CopyOrCreate WorkItem ( const CopyOrCreate WorkItem & ) = delete ;
CopyWorkItem & operator = ( const CopyWorkItem & ) = delete ;
CopyOrCreate WorkItem & operator = ( const CopyOrCreate WorkItem & ) = delete ;
CopyWorkItem ( CopyWorkItem & & o ) ROCKSDB_NOEXCEPT { * this = std : : move ( o ) ; }
CopyOrCreateWorkItem ( CopyOrCreateWorkItem & & o ) ROCKSDB_NOEXCEPT {
* this = std : : move ( o ) ;
}
CopyWorkItem & operator = ( CopyWorkItem & & o ) ROCKSDB_NOEXCEPT {
CopyOrCreate WorkItem & operator = ( CopyOrCreate WorkItem & & o ) ROCKSDB_NOEXCEPT {
src_path = std : : move ( o . src_path ) ;
src_path = std : : move ( o . src_path ) ;
dst_path = std : : move ( o . dst_path ) ;
dst_path = std : : move ( o . dst_path ) ;
contents = std : : move ( o . contents ) ;
src_env = o . src_env ;
src_env = o . src_env ;
dst_env = o . dst_env ;
dst_env = o . dst_env ;
sync = o . sync ;
sync = o . sync ;
@ -316,12 +331,14 @@ class BackupEngineImpl : public BackupEngine {
return * this ;
return * this ;
}
}
CopyWorkItem ( std : : string _src_path , std : : string _dst_path , Env * _src_env ,
CopyOrCreateWorkItem ( std : : string _src_path , std : : string _dst_path ,
Env * _dst_env , bool _sync , RateLimiter * _rate_limiter ,
std : : string _contents , Env * _src_env , Env * _dst_env ,
uint64_t _size_limit ,
bool _sync , RateLimiter * _rate_limiter ,
std : : function < void ( ) > _progress_callback = [ ] ( ) { } )
uint64_t _size_limit ,
std : : function < void ( ) > _progress_callback = [ ] ( ) { } )
: src_path ( std : : move ( _src_path ) ) ,
: src_path ( std : : move ( _src_path ) ) ,
dst_path ( std : : move ( _dst_path ) ) ,
dst_path ( std : : move ( _dst_path ) ) ,
contents ( std : : move ( _contents ) ) ,
src_env ( _src_env ) ,
src_env ( _src_env ) ,
dst_env ( _dst_env ) ,
dst_env ( _dst_env ) ,
sync ( _sync ) ,
sync ( _sync ) ,
@ -330,21 +347,23 @@ class BackupEngineImpl : public BackupEngine {
progress_callback ( _progress_callback ) { }
progress_callback ( _progress_callback ) { }
} ;
} ;
struct BackupAfterCopyWorkItem {
struct BackupAfterCopyOrCreate WorkItem {
std : : future < CopyResult > result ;
std : : future < CopyOrCreate Result > result ;
bool shared ;
bool shared ;
bool needed_to_copy ;
bool needed_to_copy ;
Env * backup_env ;
Env * backup_env ;
std : : string dst_path_tmp ;
std : : string dst_path_tmp ;
std : : string dst_path ;
std : : string dst_path ;
std : : string dst_relative ;
std : : string dst_relative ;
BackupAfterCopyWorkItem ( ) { }
BackupAfterCopyOrCreate WorkItem ( ) { }
BackupAfterCopyWorkItem ( BackupAfterCopyWorkItem & & o ) ROCKSDB_NOEXCEPT {
BackupAfterCopyOrCreateWorkItem ( BackupAfterCopyOrCreateWorkItem & & o )
ROCKSDB_NOEXCEPT {
* this = std : : move ( o ) ;
* this = std : : move ( o ) ;
}
}
BackupAfterCopyWorkItem & operator = ( BackupAfterCopyWorkItem & & o ) ROCKSDB_NOEXCEPT {
BackupAfterCopyOrCreateWorkItem & operator = (
BackupAfterCopyOrCreateWorkItem & & o ) ROCKSDB_NOEXCEPT {
result = std : : move ( o . result ) ;
result = std : : move ( o . result ) ;
shared = o . shared ;
shared = o . shared ;
needed_to_copy = o . needed_to_copy ;
needed_to_copy = o . needed_to_copy ;
@ -355,10 +374,11 @@ class BackupEngineImpl : public BackupEngine {
return * this ;
return * this ;
}
}
BackupAfterCopyWorkItem ( std : : future < CopyResult > & & _result , bool _shared ,
BackupAfterCopyOrCreateWorkItem ( std : : future < CopyOrCreateResult > & & _result ,
bool _needed_to_copy , Env * _backup_env ,
bool _shared , bool _needed_to_copy ,
std : : string _dst_path_tmp , std : : string _dst_path ,
Env * _backup_env , std : : string _dst_path_tmp ,
std : : string _dst_relative )
std : : string _dst_path ,
std : : string _dst_relative )
: result ( std : : move ( _result ) ) ,
: result ( std : : move ( _result ) ) ,
shared ( _shared ) ,
shared ( _shared ) ,
needed_to_copy ( _needed_to_copy ) ,
needed_to_copy ( _needed_to_copy ) ,
@ -368,18 +388,20 @@ class BackupEngineImpl : public BackupEngine {
dst_relative ( std : : move ( _dst_relative ) ) { }
dst_relative ( std : : move ( _dst_relative ) ) { }
} ;
} ;
struct RestoreAfterCopyWorkItem {
struct RestoreAfterCopyOrCreate WorkItem {
std : : future < CopyResult > result ;
std : : future < CopyOrCreate Result > result ;
uint32_t checksum_value ;
uint32_t checksum_value ;
RestoreAfterCopyWorkItem ( ) { }
RestoreAfterCopyOrCreate WorkItem ( ) { }
RestoreAfterCopyWorkItem ( std : : future < CopyResult > & & _result ,
RestoreAfterCopyOrCreate WorkItem ( std : : future < CopyOrCreate Result > & & _result ,
uint32_t _checksum_value )
uint32_t _checksum_value )
: result ( std : : move ( _result ) ) , checksum_value ( _checksum_value ) { }
: result ( std : : move ( _result ) ) , checksum_value ( _checksum_value ) { }
RestoreAfterCopyWorkItem ( RestoreAfterCopyWorkItem & & o ) ROCKSDB_NOEXCEPT {
RestoreAfterCopyOrCreateWorkItem ( RestoreAfterCopyOrCreateWorkItem & & o )
ROCKSDB_NOEXCEPT {
* this = std : : move ( o ) ;
* this = std : : move ( o ) ;
}
}
RestoreAfterCopyWorkItem & operator = ( RestoreAfterCopyWorkItem & & o ) ROCKSDB_NOEXCEPT {
RestoreAfterCopyOrCreateWorkItem & operator = (
RestoreAfterCopyOrCreateWorkItem & & o ) ROCKSDB_NOEXCEPT {
result = std : : move ( o . result ) ;
result = std : : move ( o . result ) ;
checksum_value = o . checksum_value ;
checksum_value = o . checksum_value ;
return * this ;
return * this ;
@ -388,17 +410,27 @@ class BackupEngineImpl : public BackupEngine {
bool initialized_ ;
bool initialized_ ;
std : : mutex byte_report_mutex_ ;
std : : mutex byte_report_mutex_ ;
channel < CopyWorkItem > files_to_copy_ ;
channel < CopyOrCreate WorkItem > files_to_copy_or_create _ ;
std : : vector < std : : thread > threads_ ;
std : : vector < std : : thread > threads_ ;
// Adds a file to the backup work queue to be copied or created if it doesn't
// already exist.
//
// Exactly one of src_dir and contents must be non-empty.
//
// @param src_dir If non-empty, the file in this directory named fname will be
// copied.
// @param fname Name of destination file and, in case of copy, source file.
// @param contents If non-empty, the file will be created with these contents.
Status AddBackupFileWorkItem (
Status AddBackupFileWorkItem (
std : : unordered_set < std : : string > & live_dst_paths ,
std : : unordered_set < std : : string > & live_dst_paths ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
std : : vector < BackupAfterCopyOrCreate WorkItem > & backup_items_to_finish ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
const std : : string & src_fname , // starts with "/"
const std : : string & fname , // starts with "/"
RateLimiter * rate_limiter , uint64_t size_limit = 0 ,
RateLimiter * rate_limiter , uint64_t size_limit = 0 ,
bool shared_checksum = false ,
bool shared_checksum = false ,
std : : function < void ( ) > progress_callback = [ ] ( ) { } ) ;
std : : function < void ( ) > progress_callback = [ ] ( ) { } ,
const std : : string & contents = std : : string ( ) ) ;
// backup state data
// backup state data
BackupID latest_backup_id_ ;
BackupID latest_backup_id_ ;
@ -451,7 +483,7 @@ BackupEngineImpl::BackupEngineImpl(Env* db_env,
read_only_ ( read_only ) { }
read_only_ ( read_only ) { }
BackupEngineImpl : : ~ BackupEngineImpl ( ) {
BackupEngineImpl : : ~ BackupEngineImpl ( ) {
files_to_copy_ . sendEof ( ) ;
files_to_copy_or_create_ . sendEof ( ) ;
for ( auto & t : threads_ ) {
for ( auto & t : threads_ ) {
t . join ( ) ;
t . join ( ) ;
}
}
@ -571,17 +603,18 @@ Status BackupEngineImpl::Initialize() {
}
}
}
}
// set up threads perform copies from files_to_copy_ in the background
// set up threads perform copies from files_to_copy_or_create_ in the
// background
for ( int t = 0 ; t < options_ . max_background_operations ; t + + ) {
for ( int t = 0 ; t < options_ . max_background_operations ; t + + ) {
threads_ . emplace_back ( [ & ] ( ) {
threads_ . emplace_back ( [ & ] ( ) {
CopyWorkItem work_item ;
CopyOrCreate WorkItem work_item ;
while ( files_to_copy_ . read ( work_item ) ) {
while ( files_to_copy_or_create_ . read ( work_item ) ) {
CopyResult result ;
CopyOrCreate Result result ;
result . status =
result . status = CopyOrCreateFile (
CopyFile ( work_item . src_path , work_item . dst_path , work_item . src_env ,
work_item . src_path , work_item . dst_path , work_item . contents ,
work_item . dst_env , work_item . sync , work_item . rate_limiter ,
work_item . src_env , work_item . dst_env , work_item . sync ,
& result . size , & result . checksum_value , work_item . size_limit ,
work_item . rate_limiter , & result . size , & result . checksum_value ,
work_item . progress_callback ) ;
work_item . size_limit , work_item . progress_callback ) ;
work_item . result . set_value ( std : : move ( result ) ) ;
work_item . result . set_value ( std : : move ( result ) ) ;
}
}
} ) ;
} ) ;
@ -616,6 +649,8 @@ Status BackupEngineImpl::CreateNewBackup(
db - > EnableFileDeletions ( false ) ;
db - > EnableFileDeletions ( false ) ;
return s ;
return s ;
}
}
TEST_SYNC_POINT ( " BackupEngineImpl::CreateNewBackup:SavedLiveFiles1 " ) ;
TEST_SYNC_POINT ( " BackupEngineImpl::CreateNewBackup:SavedLiveFiles2 " ) ;
BackupID new_backup_id = latest_backup_id_ + 1 ;
BackupID new_backup_id = latest_backup_id_ + 1 ;
assert ( backups_ . find ( new_backup_id ) = = backups_ . end ( ) ) ;
assert ( backups_ . find ( new_backup_id ) = = backups_ . end ( ) ) ;
@ -650,8 +685,9 @@ Status BackupEngineImpl::CreateNewBackup(
std : : unordered_set < std : : string > live_dst_paths ;
std : : unordered_set < std : : string > live_dst_paths ;
live_dst_paths . reserve ( live_files . size ( ) + live_wal_files . size ( ) ) ;
live_dst_paths . reserve ( live_files . size ( ) + live_wal_files . size ( ) ) ;
std : : vector < BackupAfterCopyWorkItem > backup_items_to_finish ;
std : : vector < BackupAfterCopyOrCreateWorkItem > backup_items_to_finish ;
// Add a CopyWorkItem to the channel for each live file
// Add a CopyOrCreateWorkItem to the channel for each live file
std : : string manifest_fname , current_fname ;
for ( size_t i = 0 ; s . ok ( ) & & i < live_files . size ( ) ; + + i ) {
for ( size_t i = 0 ; s . ok ( ) & & i < live_files . size ( ) ; + + i ) {
uint64_t number ;
uint64_t number ;
FileType type ;
FileType type ;
@ -663,6 +699,15 @@ Status BackupEngineImpl::CreateNewBackup(
// we should only get sst, manifest and current files here
// we should only get sst, manifest and current files here
assert ( type = = kTableFile | | type = = kDescriptorFile | |
assert ( type = = kTableFile | | type = = kDescriptorFile | |
type = = kCurrentFile ) ;
type = = kCurrentFile ) ;
if ( type = = kCurrentFile ) {
// We will craft the current file manually to ensure it's consistent with
// the manifest number. This is necessary because current's file contents
// can change during backup.
current_fname = live_files [ i ] ;
continue ;
} else if ( type = = kDescriptorFile ) {
manifest_fname = live_files [ i ] ;
}
// rules:
// rules:
// * if it's kTableFile, then it's shared
// * if it's kTableFile, then it's shared
@ -675,7 +720,15 @@ Status BackupEngineImpl::CreateNewBackup(
options_ . share_files_with_checksum & & type = = kTableFile ,
options_ . share_files_with_checksum & & type = = kTableFile ,
progress_callback ) ;
progress_callback ) ;
}
}
// Add a CopyWorkItem to the channel for each WAL file
if ( s . ok ( ) & & ! current_fname . empty ( ) & & ! manifest_fname . empty ( ) ) {
// Write the current file with the manifest filename as its contents.
s = AddBackupFileWorkItem (
live_dst_paths , backup_items_to_finish , new_backup_id ,
false /* shared */ , " " /* src_dir */ , CurrentFileName ( " " ) ,
rate_limiter . get ( ) , 0 /* size_limit */ , false /* shared_checksum */ ,
progress_callback , manifest_fname . substr ( 1 ) + " \n " ) ;
}
// Add a CopyOrCreateWorkItem to the channel for each WAL file
for ( size_t i = 0 ; s . ok ( ) & & i < live_wal_files . size ( ) ; + + i ) {
for ( size_t i = 0 ; s . ok ( ) & & i < live_wal_files . size ( ) ; + + i ) {
if ( live_wal_files [ i ] - > Type ( ) = = kAliveLogFile ) {
if ( live_wal_files [ i ] - > Type ( ) = = kAliveLogFile ) {
// we only care about live log files
// we only care about live log files
@ -938,7 +991,7 @@ Status BackupEngineImpl::RestoreDBFromBackup(
copy_file_buffer_size_ = rate_limiter - > GetSingleBurstBytes ( ) ;
copy_file_buffer_size_ = rate_limiter - > GetSingleBurstBytes ( ) ;
}
}
Status s ;
Status s ;
std : : vector < RestoreAfterCopyWorkItem > restore_items_to_finish ;
std : : vector < RestoreAfterCopyOrCreate WorkItem > restore_items_to_finish ;
for ( const auto & file_info : backup - > GetFiles ( ) ) {
for ( const auto & file_info : backup - > GetFiles ( ) ) {
const std : : string & file = file_info - > filename ;
const std : : string & file = file_info - > filename ;
std : : string dst ;
std : : string dst ;
@ -968,18 +1021,15 @@ Status BackupEngineImpl::RestoreDBFromBackup(
" / " + dst ;
" / " + dst ;
Log ( options_ . info_log , " Restoring %s to %s \n " , file . c_str ( ) , dst . c_str ( ) ) ;
Log ( options_ . info_log , " Restoring %s to %s \n " , file . c_str ( ) , dst . c_str ( ) ) ;
CopyWorkItem copy_work_item ( GetAbsolutePath ( file ) ,
CopyOrCreateWorkItem copy_or_create_work_item (
dst ,
GetAbsolutePath ( file ) , dst , " " /* contents */ , backup_env_ , db_env_ ,
backup_env_ ,
false , rate_limiter . get ( ) , 0 /* size_limit */ ) ;
db_env_ ,
RestoreAfterCopyOrCreateWorkItem after_copy_or_create_work_item (
false ,
copy_or_create_work_item . result . get_future ( ) ,
rate_limiter . get ( ) ,
file_info - > checksum_value ) ;
0 /* size_limit */ ) ;
files_to_copy_or_create_ . write ( std : : move ( copy_or_create_work_item ) ) ;
RestoreAfterCopyWorkItem after_copy_work_item (
restore_items_to_finish . push_back (
copy_work_item . result . get_future ( ) ,
std : : move ( after_copy_or_create_work_item ) ) ;
file_info - > checksum_value ) ;
files_to_copy_ . write ( std : : move ( copy_work_item ) ) ;
restore_items_to_finish . push_back ( std : : move ( after_copy_work_item ) ) ;
}
}
Status item_status ;
Status item_status ;
for ( auto & item : restore_items_to_finish ) {
for ( auto & item : restore_items_to_finish ) {
@ -1078,12 +1128,12 @@ Status BackupEngineImpl::PutLatestBackupFileContents(uint32_t latest_backup) {
return s ;
return s ;
}
}
Status BackupEngineImpl : : CopyFile ( const std : : string & src ,
Status BackupEngineImpl : : CopyOrCreate File (
const std : : string & dst , Env * src_env ,
const std : : string & src , const std : : string & dst , const std : : string & contents ,
Env * dst_env , bool sync ,
Env * src_env , Env * dst_env , bool sync , RateLimiter * rate_limiter ,
RateLimiter * rate_limiter , uint64 _t * size ,
uint64_t * size , uint32 _t* checksum_value , uint64_t size_limit ,
uint32_t * checksum_value , uint64_t size_limit ,
std : : function < void ( ) > progress_callback ) {
std : : function < void ( ) > progress_callback ) {
assert ( src . empty ( ) ! = contents . empty ( ) ) ;
Status s ;
Status s ;
unique_ptr < WritableFile > dst_file ;
unique_ptr < WritableFile > dst_file ;
unique_ptr < SequentialFile > src_file ;
unique_ptr < SequentialFile > src_file ;
@ -1102,9 +1152,9 @@ Status BackupEngineImpl::CopyFile(const std::string& src,
size_limit = std : : numeric_limits < uint64_t > : : max ( ) ;
size_limit = std : : numeric_limits < uint64_t > : : max ( ) ;
}
}
s = src_env - > NewSequentialFile ( src , & src _file, env_options ) ;
s = dst_env - > NewWritableFile ( dst , & dst _file, env_options ) ;
if ( s . ok ( ) ) {
if ( s . ok ( ) & & ! src . empty ( ) ) {
s = dst_env - > NewWritableFile ( dst , & dst _file, env_options ) ;
s = src_env - > NewSequentialFile ( src , & src _file, env_options ) ;
}
}
if ( ! s . ok ( ) ) {
if ( ! s . ok ( ) ) {
return s ;
return s ;
@ -1112,19 +1162,28 @@ Status BackupEngineImpl::CopyFile(const std::string& src,
unique_ptr < WritableFileWriter > dest_writer (
unique_ptr < WritableFileWriter > dest_writer (
new WritableFileWriter ( std : : move ( dst_file ) , env_options ) ) ;
new WritableFileWriter ( std : : move ( dst_file ) , env_options ) ) ;
unique_ptr < SequentialFileReader > src_reader (
unique_ptr < SequentialFileReader > src_reader ;
new SequentialFileReader ( std : : move ( src_file ) ) ) ;
unique_ptr < char [ ] > buf ;
unique_ptr < char [ ] > buf ( new char [ copy_file_buffer_size_ ] ) ;
if ( ! src . empty ( ) ) {
Slice data ;
src_reader . reset ( new SequentialFileReader ( std : : move ( src_file ) ) ) ;
buf . reset ( new char [ copy_file_buffer_size_ ] ) ;
}
Slice data ;
uint64_t processed_buffer_size = 0 ;
uint64_t processed_buffer_size = 0 ;
do {
do {
if ( stop_backup_ . load ( std : : memory_order_acquire ) ) {
if ( stop_backup_ . load ( std : : memory_order_acquire ) ) {
return Status : : Incomplete ( " Backup stopped " ) ;
return Status : : Incomplete ( " Backup stopped " ) ;
}
}
size_t buffer_to_read = ( copy_file_buffer_size_ < size_limit ) ?
if ( ! src . empty ( ) ) {
copy_file_buffer_size_ : size_limit ;
size_t buffer_to_read = ( copy_file_buffer_size_ < size_limit )
s = src_reader - > Read ( buffer_to_read , & data , buf . get ( ) ) ;
? copy_file_buffer_size_
: size_limit ;
s = src_reader - > Read ( buffer_to_read , & data , buf . get ( ) ) ;
processed_buffer_size + = buffer_to_read ;
} else {
data = contents ;
}
size_limit - = data . size ( ) ;
size_limit - = data . size ( ) ;
if ( ! s . ok ( ) ) {
if ( ! s . ok ( ) ) {
@ -1135,38 +1194,38 @@ Status BackupEngineImpl::CopyFile(const std::string& src,
* size + = data . size ( ) ;
* size + = data . size ( ) ;
}
}
if ( checksum_value ! = nullptr ) {
if ( checksum_value ! = nullptr ) {
* checksum_value = crc32c : : Extend ( * checksum_value , data . data ( ) ,
* checksum_value =
data . size ( ) ) ;
crc32c : : Extend ( * checksum_value , data . data ( ) , data . size ( ) ) ;
}
}
s = dest_writer - > Append ( data ) ;
s = dest_writer - > Append ( data ) ;
if ( rate_limiter ! = nullptr ) {
if ( rate_limiter ! = nullptr ) {
rate_limiter - > Request ( data . size ( ) , Env : : IO_LOW ) ;
rate_limiter - > Request ( data . size ( ) , Env : : IO_LOW ) ;
}
}
processed_buffer_size + = buffer_to_read ;
if ( processed_buffer_size > options_ . callback_trigger_interval_size ) {
if ( processed_buffer_size > options_ . callback_trigger_interval_size ) {
processed_buffer_size - = options_ . callback_trigger_interval_size ;
processed_buffer_size - = options_ . callback_trigger_interval_size ;
std : : lock_guard < std : : mutex > lock ( byte_report_mutex_ ) ;
std : : lock_guard < std : : mutex > lock ( byte_report_mutex_ ) ;
progress_callback ( ) ;
progress_callback ( ) ;
}
}
} while ( s . ok ( ) & & data . size ( ) > 0 & & size_limit > 0 ) ;
} while ( s . ok ( ) & & contents . empty ( ) & & data . size ( ) > 0 & & size_limit > 0 ) ;
if ( s . ok ( ) & & sync ) {
if ( s . ok ( ) & & sync ) {
s = dest_writer - > Sync ( false ) ;
s = dest_writer - > Sync ( false ) ;
}
}
return s ;
return s ;
}
}
// src_ fname will always start with "/"
// fname will always start with "/"
Status BackupEngineImpl : : AddBackupFileWorkItem (
Status BackupEngineImpl : : AddBackupFileWorkItem (
std : : unordered_set < std : : string > & live_dst_paths ,
std : : unordered_set < std : : string > & live_dst_paths ,
std : : vector < BackupAfterCopyWorkItem > & backup_items_to_finish ,
std : : vector < BackupAfterCopyOrCreate WorkItem > & backup_items_to_finish ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
BackupID backup_id , bool shared , const std : : string & src_dir ,
const std : : string & src_fname , RateLimiter * rate_limiter ,
const std : : string & fname , RateLimiter * rate_limiter , uint64_t size_limit ,
uint64_t size_limit , bool shared_checksum ,
bool shared_checksum , std : : function < void ( ) > progress_callback ,
std : : function < void ( ) > progress_callback ) {
const std : : string & contents ) {
assert ( src_fname . size ( ) > 0 & & src_fname [ 0 ] = = ' / ' ) ;
assert ( ! fname . empty ( ) & & fname [ 0 ] = = ' / ' ) ;
std : : string dst_relative = src_fname . substr ( 1 ) ;
assert ( contents . empty ( ) ! = src_dir . empty ( ) ) ;
std : : string dst_relative = fname . substr ( 1 ) ;
std : : string dst_relative_tmp ;
std : : string dst_relative_tmp ;
Status s ;
Status s ;
uint64_t size ;
uint64_t size ;
@ -1174,12 +1233,10 @@ Status BackupEngineImpl::AddBackupFileWorkItem(
if ( shared & & shared_checksum ) {
if ( shared & & shared_checksum ) {
// add checksum and file length to the file name
// add checksum and file length to the file name
s = CalculateChecksum ( src_dir + src_fname ,
s = CalculateChecksum ( src_dir + fname , db_env_ , size_limit ,
db_env_ ,
size_limit ,
& checksum_value ) ;
& checksum_value ) ;
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
s = db_env_ - > GetFileSize ( src_dir + src_ fname, & size ) ;
s = db_env_ - > GetFileSize ( src_dir + fname , & size ) ;
}
}
if ( ! s . ok ( ) ) {
if ( ! s . ok ( ) ) {
return s ;
return s ;
@ -1218,12 +1275,14 @@ Status BackupEngineImpl::AddBackupFileWorkItem(
}
}
}
}
if ( shared & & ( same_path | | file_exists ) ) {
if ( ! contents . empty ( ) ) {
need_to_copy = false ;
} else if ( shared & & ( same_path | | file_exists ) ) {
need_to_copy = false ;
need_to_copy = false ;
if ( shared_checksum ) {
if ( shared_checksum ) {
Log ( options_ . info_log ,
Log ( options_ . info_log ,
" %s already present, with checksum %u and size % " PRIu64 ,
" %s already present, with checksum %u and size % " PRIu64 ,
src_ fname. c_str ( ) , checksum_value , size ) ;
fname . c_str ( ) , checksum_value , size ) ;
} else if ( backuped_file_infos_ . find ( dst_relative ) = =
} else if ( backuped_file_infos_ . find ( dst_relative ) = =
backuped_file_infos_ . end ( ) & & ! same_path ) {
backuped_file_infos_ . end ( ) & & ! same_path ) {
// file already exists, but it's not referenced by any backup. overwrite
// file already exists, but it's not referenced by any backup. overwrite
@ -1231,48 +1290,39 @@ Status BackupEngineImpl::AddBackupFileWorkItem(
Log ( options_ . info_log ,
Log ( options_ . info_log ,
" %s already present, but not referenced by any backup. We will "
" %s already present, but not referenced by any backup. We will "
" overwrite the file. " ,
" overwrite the file. " ,
src_ fname. c_str ( ) ) ;
fname . c_str ( ) ) ;
need_to_copy = true ;
need_to_copy = true ;
backup_env_ - > DeleteFile ( dst_path ) ;
backup_env_ - > DeleteFile ( dst_path ) ;
} else {
} else {
// the file is present and referenced by a backup
// the file is present and referenced by a backup
db_env_ - > GetFileSize ( src_dir + src_ fname, & size ) ; // Ignore error
db_env_ - > GetFileSize ( src_dir + fname , & size ) ; // Ignore error
Log ( options_ . info_log , " %s already present, calculate checksum " ,
Log ( options_ . info_log , " %s already present, calculate checksum " ,
src_ fname. c_str ( ) ) ;
fname . c_str ( ) ) ;
s = CalculateChecksum ( src_dir + src_ fname, db_env_ , size_limit ,
s = CalculateChecksum ( src_dir + fname , db_env_ , size_limit ,
& checksum_value ) ;
& checksum_value ) ;
}
}
}
}
live_dst_paths . insert ( dst_path ) ;
live_dst_paths . insert ( dst_path ) ;
if ( need_to_copy ) {
if ( ! contents . empty ( ) | | need_to_copy ) {
Log ( options_ . info_log , " Copying %s to %s " , src_fname . c_str ( ) ,
Log ( options_ . info_log , " Copying %s to %s " , fname . c_str ( ) ,
dst_path_tmp . c_str ( ) ) ;
dst_path_tmp . c_str ( ) ) ;
CopyWorkItem copy_work_item ( src_dir + src_fname , dst_path_tmp , db_env_ ,
CopyOrCreateWorkItem copy_or_create_work_item (
backup_env_ , options_ . sync , rate_limiter ,
src_dir . empty ( ) ? " " : src_dir + fname , dst_path_tmp , contents , db_env_ ,
size_limit , progress_callback ) ;
backup_env_ , options_ . sync , rate_limiter , size_limit ,
BackupAfterCopyWorkItem after_copy_work_item (
progress_callback ) ;
copy_work_item . result . get_future ( ) ,
BackupAfterCopyOrCreateWorkItem after_copy_or_create_work_item (
shared ,
copy_or_create_work_item . result . get_future ( ) , shared , need_to_copy ,
need_to_copy ,
backup_env_ , dst_path_tmp , dst_path , dst_relative ) ;
backup_env_ ,
files_to_copy_or_create_ . write ( std : : move ( copy_or_create_work_item ) ) ;
dst_path_tmp ,
backup_items_to_finish . push_back ( std : : move ( after_copy_or_create_work_item ) ) ;
dst_path ,
dst_relative ) ;
files_to_copy_ . write ( std : : move ( copy_work_item ) ) ;
backup_items_to_finish . push_back ( std : : move ( after_copy_work_item ) ) ;
} else {
} else {
std : : promise < CopyResult > promise_result ;
std : : promise < CopyOrCreateResult > promise_result ;
BackupAfterCopyWorkItem after_copy_work_item (
BackupAfterCopyOrCreateWorkItem after_copy_or_create_work_item (
promise_result . get_future ( ) ,
promise_result . get_future ( ) , shared , need_to_copy , backup_env_ ,
shared ,
dst_path_tmp , dst_path , dst_relative ) ;
need_to_copy ,
backup_items_to_finish . push_back ( std : : move ( after_copy_or_create_work_item ) ) ;
backup_env_ ,
CopyOrCreateResult result ;
dst_path_tmp ,
dst_path ,
dst_relative ) ;
backup_items_to_finish . push_back ( std : : move ( after_copy_work_item ) ) ;
CopyResult result ;
result . status = s ;
result . status = s ;
result . size = size ;
result . size = size ;
result . checksum_value = checksum_value ;
result . checksum_value = checksum_value ;