@ -113,159 +113,162 @@ TEST_F(WriteCallbackTest, WriteWithCallbackTest) {
for ( auto & allow_parallel : { true , false } ) {
for ( auto & allow_parallel : { true , false } ) {
for ( auto & allow_batching : { true , false } ) {
for ( auto & allow_batching : { true , false } ) {
for ( auto & write_group : write_scenarios ) {
for ( auto & enable_WAL : { true , false } ) {
Options options ;
for ( auto & write_group : write_scenarios ) {
options . create_if_missing = true ;
Options options ;
options . allow_concurrent_memtable_write = allow_parallel ;
options . create_if_missing = true ;
options . allow_concurrent_memtable_write = allow_parallel ;
WriteOptions write_options ;
ReadOptions read_options ;
ReadOptions read_options ;
DB * db ;
DB * db ;
DBImpl * db_impl ;
DBImpl * db_impl ;
ASSERT_OK ( DB : : Open ( options , dbname , & db ) ) ;
ASSERT_OK ( DB : : Open ( options , dbname , & db ) ) ;
db_impl = dynamic_cast < DBImpl * > ( db ) ;
db_impl = dynamic_cast < DBImpl * > ( db ) ;
ASSERT_TRUE ( db_impl ) ;
ASSERT_TRUE ( db_impl ) ;
std : : atomic < uint64_t > threads_waiting ( 0 ) ;
std : : atomic < uint64_t > seq ( db_impl - > GetLatestSequenceNumber ( ) ) ;
ASSERT_EQ ( db_impl - > GetLatestSequenceNumber ( ) , 0 ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
" WriteThread::JoinBatchGroup:Wait " , [ & ] ( void * arg ) {
uint64_t cur_threads_waiting = 0 ;
bool is_leader = false ;
bool is_last = false ;
// who am i
do {
cur_threads_waiting = threads_waiting . load ( ) ;
is_leader = ( cur_threads_waiting = = 0 ) ;
is_last = ( cur_threads_waiting = = write_group . size ( ) - 1 ) ;
} while ( ! threads_waiting . compare_exchange_strong (
cur_threads_waiting , cur_threads_waiting + 1 ) ) ;
// check my state
auto * writer = reinterpret_cast < WriteThread : : Writer * > ( arg ) ;
if ( is_leader ) {
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_GROUP_LEADER ) ;
} else {
ASSERT_TRUE ( writer - > state = = WriteThread : : State : : STATE_INIT ) ;
}
// (meta test) the first WriteOP should indeed be the first
// and the last should be the last (all others can be out of
// order)
if ( is_leader ) {
ASSERT_TRUE ( writer - > callback - > Callback ( nullptr ) . ok ( ) = =
! write_group . front ( ) . callback_ . should_fail_ ) ;
} else if ( is_last ) {
ASSERT_TRUE ( writer - > callback - > Callback ( nullptr ) . ok ( ) = =
! write_group . back ( ) . callback_ . should_fail_ ) ;
}
// wait for friends
while ( threads_waiting . load ( ) < write_group . size ( ) ) {
}
} ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
" WriteThread::JoinBatchGroup:DoneWaiting " , [ & ] ( void * arg ) {
// check my state
auto * writer = reinterpret_cast < WriteThread : : Writer * > ( arg ) ;
if ( ! allow_batching ) {
// no batching so everyone should be a leader
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_GROUP_LEADER ) ;
} else if ( ! allow_parallel ) {
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_COMPLETED ) ;
}
} ) ;
std : : atomic < uint32_t > thread_num ( 0 ) ;
std : : atomic < char > dummy_key ( 0 ) ;
std : : function < void ( ) > write_with_callback_func = [ & ] ( ) {
uint32_t i = thread_num . fetch_add ( 1 ) ;
Random rnd ( i ) ;
// leaders gotta lead
while ( i > 0 & & threads_waiting . load ( ) < 1 ) {
}
std : : atomic < uint64_t > threads_waiting ( 0 ) ;
// loser has to lose
std : : atomic < uint64_t > seq ( db_impl - > GetLatestSequenceNumber ( ) ) ;
while ( i = = write_group . size ( ) - 1 & &
ASSERT_EQ ( db_impl - > GetLatestSequenceNumber ( ) , 0 ) ;
threads_waiting . load ( ) < write_group . size ( ) - 1 ) {
}
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
auto & write_op = write_group . at ( i ) ;
" WriteThread::JoinBatchGroup:Wait " , [ & ] ( void * arg ) {
write_op . Clear ( ) ;
uint64_t cur_threads_waiting = 0 ;
write_op . callback_ . allow_batching_ = allow_batching ;
bool is_leader = false ;
bool is_last = false ;
// who am i
// insert some keys
for ( uint32_t j = 0 ; j < rnd . Next ( ) % 50 ; j + + ) {
// grab unique key
char my_key = 0 ;
do {
do {
cur_threads_waiting = threads_waiting . load ( ) ;
my_key = dummy_key . load ( ) ;
is_leader = ( cur_threads_waiting = = 0 ) ;
} while ( ! dummy_key . compare_exchange_strong ( my_key , my_key + 1 ) ) ;
is_last = ( cur_threads_waiting = = write_group . size ( ) - 1 ) ;
} while ( ! threads_waiting . compare_exchange_strong (
cur_threads_waiting , cur_threads_waiting + 1 ) ) ;
// check my state
auto * writer = reinterpret_cast < WriteThread : : Writer * > ( arg ) ;
if ( is_leader ) {
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_GROUP_LEADER ) ;
} else {
ASSERT_TRUE ( writer - > state = = WriteThread : : State : : STATE_INIT ) ;
}
// (meta test) the first WriteOP should indeed be the first
string skey ( 5 , my_key ) ;
// and the last should be the last (all others can be out of
string sval ( 10 , my_key ) ;
// order)
write_op . Put ( skey , sval ) ;
if ( is_leader ) {
ASSERT_TRUE ( writer - > callback - > Callback ( nullptr ) . ok ( ) = =
! write_group . front ( ) . callback_ . should_fail_ ) ;
} else if ( is_last ) {
ASSERT_TRUE ( writer - > callback - > Callback ( nullptr ) . ok ( ) = =
! write_group . back ( ) . callback_ . should_fail_ ) ;
}
// wait for friends
if ( ! write_op . callback_ . should_fail_ ) {
while ( threads_waiting . load ( ) < write_group . size ( ) ) {
seq . fetch_add ( 1 ) ;
}
}
} ) ;
}
rocksdb : : SyncPoint : : GetInstance ( ) - > SetCallBack (
" WriteThread::JoinBatchGroup:DoneWaiting " , [ & ] ( void * arg ) {
// check my state
auto * writer = reinterpret_cast < WriteThread : : Writer * > ( arg ) ;
if ( ! allow_batching ) {
// no batching so everyone should be a leader
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_GROUP_LEADER ) ;
} else if ( ! allow_parallel ) {
ASSERT_TRUE ( writer - > state = =
WriteThread : : State : : STATE_COMPLETED ) ;
}
} ) ;
std : : atomic < uint32_t > thread_num ( 0 ) ;
std : : atomic < char > dummy_key ( 0 ) ;
std : : function < void ( ) > write_with_callback_func = [ & ] ( ) {
uint32_t i = thread_num . fetch_add ( 1 ) ;
Random rnd ( i ) ;
// leaders gotta lead
while ( i > 0 & & threads_waiting . load ( ) < 1 ) {
}
// loser has to lose
while ( i = = write_group . size ( ) - 1 & &
threads_waiting . load ( ) < write_group . size ( ) - 1 ) {
}
auto & write_op = write_group . at ( i ) ;
write_op . Clear ( ) ;
write_op . callback_ . allow_batching_ = allow_batching ;
// insert some keys
for ( uint32_t j = 0 ; j < rnd . Next ( ) % 50 ; j + + ) {
// grab unique key
char my_key = 0 ;
do {
my_key = dummy_key . load ( ) ;
} while ( ! dummy_key . compare_exchange_strong ( my_key , my_key + 1 ) ) ;
string skey ( 5 , my_key ) ;
WriteOptions woptions ;
string sval ( 10 , my_key ) ;
woptions . disableWAL = ! enable_WAL ;
write_op . Put ( skey , sval ) ;
woptions . sync = enable_WAL ;
Status s = db_impl - > WriteWithCallback (
woptions , & write_op . write_batch_ , & write_op . callback_ ) ;
if ( ! write_op . callback_ . should_fail_ ) {
if ( write_op . callback_ . should_fail_ ) {
seq . fetch_add ( 1 ) ;
ASSERT_TRUE ( s . IsBusy ( ) ) ;
} else {
ASSERT_OK ( s ) ;
}
}
}
} ;
WriteOptions woptions ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
Status s = db_impl - > WriteWithCallback (
woptions , & write_op . write_batch_ , & write_op . callback_ ) ;
if ( write_op . callback_ . should_fail_ ) {
// do all the writes
ASSERT_TRUE ( s . IsBusy ( ) ) ;
std : : vector < std : : thread > threads ;
} else {
for ( uint32_t i = 0 ; i < write_group . size ( ) ; i + + ) {
ASSERT_OK ( s ) ;
threads . emplace_back ( write_with_callback_func ) ;
}
for ( auto & t : threads ) {
t . join ( ) ;
}
}
} ;
rocksdb : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;
// do all the writes
std : : vector < std : : thread > threads ;
for ( uint32_t i = 0 ; i < write_group . size ( ) ; i + + ) {
threads . emplace_back ( write_with_callback_func ) ;
}
for ( auto & t : threads ) {
t . join ( ) ;
}
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
rocksdb : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;
// check for keys
// check for keys
string value ;
string value ;
for ( auto & w : write_group ) {
for ( auto & w : write_group ) {
ASSERT_TRUE ( w . callback_ . was_called_ ) ;
ASSERT_TRUE ( w . callback_ . was_called_ ) ;
for ( auto & kvp : w . kvs_ ) {
for ( auto & kvp : w . kvs_ ) {
if ( w . callback_ . should_fail_ ) {
if ( w . callback_ . should_fail_ ) {
ASSERT_TRUE (
ASSERT_TRUE (
db - > Get ( read_options , kvp . first , & value ) . IsNotFound ( ) ) ;
db - > Get ( read_options , kvp . first , & value ) . IsNotFound ( ) ) ;
} else {
} else {
ASSERT_OK ( db - > Get ( read_options , kvp . first , & value ) ) ;
ASSERT_OK ( db - > Get ( read_options , kvp . first , & value ) ) ;
ASSERT_EQ ( value , kvp . second ) ;
ASSERT_EQ ( value , kvp . second ) ;
}
}
}
}
}
}
ASSERT_EQ ( seq . load ( ) , db_impl - > GetLatestSequenceNumber ( ) ) ;
ASSERT_EQ ( seq . load ( ) , db_impl - > GetLatestSequenceNumber ( ) ) ;
delete db ;
delete db ;
DestroyDB ( dbname , options ) ;
DestroyDB ( dbname , options ) ;
}
}
}
}
}
}
}