@ -14,46 +14,65 @@
namespace rocksdb {
namespace rocksdb {
class Sleeping BackgroundTask {
class Simulated BackgroundTask {
public :
public :
SleepingBackgroundTask ( const void * db_key , const std : : string & db_name ,
SimulatedBackgroundTask (
const void * cf_key , const std : : string & cf_name )
const void * db_key , const std : : string & db_name ,
const void * cf_key , const std : : string & cf_name ,
const ThreadStatus : : OperationType operation_type =
ThreadStatus : : OP_UNKNOWN ,
const ThreadStatus : : StateType state_type =
ThreadStatus : : STATE_UNKNOWN )
: db_key_ ( db_key ) , db_name_ ( db_name ) ,
: db_key_ ( db_key ) , db_name_ ( db_name ) ,
cf_key_ ( cf_key ) , cf_name_ ( cf_name ) ,
cf_key_ ( cf_key ) , cf_name_ ( cf_name ) ,
should_sleep_ ( true ) , sleeping_count_ ( 0 ) {
operation_type_ ( operation_type ) , state_type_ ( state_type ) ,
should_run_ ( true ) , running_count_ ( 0 ) {
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > NewColumnFamilyInfo (
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > NewColumnFamilyInfo (
db_key_ , db_name_ , cf_key_ , cf_name_ ) ;
db_key_ , db_name_ , cf_key_ , cf_name_ ) ;
}
}
~ Sleeping BackgroundTask ( ) {
~ Simulated BackgroundTask ( ) {
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > EraseDatabaseInfo ( db_key_ ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > EraseDatabaseInfo ( db_key_ ) ;
}
}
void DoSleep ( ) {
void Run ( ) {
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetColumnFamilyInfoKey ( cf_key_ ) ;
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
sleeping_count_ + + ;
running_count_ + + ;
while ( should_sleep_ ) {
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetColumnFamilyInfoKey ( cf_key_ ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetThreadOperation (
operation_type_ ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetThreadState ( state_type_ ) ;
while ( should_run_ ) {
bg_cv_ . wait ( l ) ;
bg_cv_ . wait ( l ) ;
}
}
sleeping_count_ - - ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > ClearThreadState ( ) ;
bg_cv_ . notify_all ( ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > ClearThreadOperation ( ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetColumnFamilyInfoKey ( 0 ) ;
Env : : Default ( ) - > GetThreadStatusUpdater ( ) - > SetColumnFamilyInfoKey ( 0 ) ;
running_count_ - - ;
bg_cv_ . notify_all ( ) ;
}
}
void WakeUp ( ) {
void FinishAllTasks ( ) {
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
should_sleep _ = false ;
should_run _ = false ;
bg_cv_ . notify_all ( ) ;
bg_cv_ . notify_all ( ) ;
}
}
void WaitUntilScheduled ( int job_count , Env * env ) {
while ( running_count_ < job_count ) {
env - > SleepForMicroseconds ( 1000 ) ;
}
}
void WaitUntilDone ( ) {
void WaitUntilDone ( ) {
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
std : : unique_lock < std : : mutex > l ( mutex_ ) ;
while ( sleeping_count_ > 0 ) {
while ( runn ing_count_ > 0 ) {
bg_cv_ . wait ( l ) ;
bg_cv_ . wait ( l ) ;
}
}
}
}
static void DoSleep Task ( void * arg ) {
static void DoSimulated Task ( void * arg ) {
reinterpret_cast < Sleeping BackgroundTask * > ( arg ) - > DoSleep ( ) ;
reinterpret_cast < Simulated BackgroundTask * > ( arg ) - > Run ( ) ;
}
}
private :
private :
@ -61,10 +80,12 @@ class SleepingBackgroundTask {
const std : : string db_name_ ;
const std : : string db_name_ ;
const void * cf_key_ ;
const void * cf_key_ ;
const std : : string cf_name_ ;
const std : : string cf_name_ ;
const ThreadStatus : : OperationType operation_type_ ;
const ThreadStatus : : StateType state_type_ ;
std : : mutex mutex_ ;
std : : mutex mutex_ ;
std : : condition_variable bg_cv_ ;
std : : condition_variable bg_cv_ ;
bool should_sleep _ ;
bool should_run _ ;
std : : atomic < int > sleep ing_count_;
std : : atomic < int > runn ing_count_;
} ;
} ;
class ThreadListTest {
class ThreadListTest {
@ -73,72 +94,232 @@ class ThreadListTest {
}
}
} ;
} ;
TEST ( ThreadListTest , EventTables ) {
// verify the global tables for operations and states are properly indexed.
for ( int type = 0 ; type ! = ThreadStatus : : NUM_OP_TYPES ; + + type ) {
ASSERT_EQ ( global_operation_table [ type ] . type , type ) ;
}
for ( int type = 0 ; type ! = ThreadStatus : : NUM_STATE_TYPES ; + + type ) {
ASSERT_EQ ( global_state_table [ type ] . type , type ) ;
}
}
TEST ( ThreadListTest , SimpleColumnFamilyInfoTest ) {
TEST ( ThreadListTest , SimpleColumnFamilyInfoTest ) {
Env * env = Env : : Default ( ) ;
Env * env = Env : : Default ( ) ;
const int kHighPriorityThreads = 3 ;
const int kHighPriorityThreads = 3 ;
const int kLowPriorityThreads = 5 ;
const int kLowPriorityThreads = 5 ;
const int kSleepingHighPriThreads = kHighPriorityThreads - 1 ;
const int kSimulated HighPriThreads = kHighPriorityThreads - 1 ;
const int kSleepingLowPriThreads = kLowPriorityThreads / 3 ;
const int kSimulated LowPriThreads = kLowPriorityThreads / 3 ;
env - > SetBackgroundThreads ( kHighPriorityThreads , Env : : HIGH ) ;
env - > SetBackgroundThreads ( kHighPriorityThreads , Env : : HIGH ) ;
env - > SetBackgroundThreads ( kLowPriorityThreads , Env : : LOW ) ;
env - > SetBackgroundThreads ( kLowPriorityThreads , Env : : LOW ) ;
SleepingBackgroundTask sleeping_task (
SimulatedBackgroundTask runn ing_task (
reinterpret_cast < void * > ( 1234 ) , " sleeping " ,
reinterpret_cast < void * > ( 1234 ) , " runn ing" ,
reinterpret_cast < void * > ( 5678 ) , " pikachu " ) ;
reinterpret_cast < void * > ( 5678 ) , " pikachu " ) ;
for ( int test = 0 ; test < kSleeping HighPriThreads ; + + test ) {
for ( int test = 0 ; test < kSimulated HighPriThreads ; + + test ) {
env - > Schedule ( & SleepingBackgroundTask : : DoSleep Task ,
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulated Task ,
& sleep ing_task, Env : : Priority : : HIGH ) ;
& runn ing_task, Env : : Priority : : HIGH ) ;
}
}
for ( int test = 0 ; test < kSleeping LowPriThreads ; + + test ) {
for ( int test = 0 ; test < kSimulated LowPriThreads ; + + test ) {
env - > Schedule ( & SleepingBackgroundTask : : DoSleep Task ,
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulated Task ,
& sleep ing_task, Env : : Priority : : LOW ) ;
& runn ing_task, Env : : Priority : : LOW ) ;
}
}
running_task . WaitUntilScheduled (
// make sure everything is scheduled.
kSimulatedHighPriThreads + kSimulatedLowPriThreads , env ) ;
env - > SleepForMicroseconds ( 10000 ) ;
std : : vector < ThreadStatus > thread_list ;
std : : vector < ThreadStatus > thread_list ;
// Verify the number of sleep ing threads in each pool.
// Verify the number of runn ing threads in each pool.
env - > GetThreadList ( & thread_list ) ;
env - > GetThreadList ( & thread_list ) ;
int sleep ing_count[ ThreadStatus : : ThreadType : : TOTAL ] = { 0 } ;
int runn ing_count[ ThreadStatus : : NUM_THREAD_TYPES ] = { 0 } ;
for ( auto thread_status : thread_list ) {
for ( auto thread_status : thread_list ) {
if ( thread_status . cf_name = = " pikachu " & &
if ( thread_status . cf_name = = " pikachu " & &
thread_status . db_name = = " sleep ing" ) {
thread_status . db_name = = " runn ing" ) {
sleep ing_count[ thread_status . thread_type ] + + ;
runn ing_count[ thread_status . thread_type ] + + ;
}
}
}
}
ASSERT_EQ (
ASSERT_EQ (
sleep ing_count[ ThreadStatus : : ThreadType : : ROCKSDB_ HIGH_PRIORITY] ,
runn ing_count[ ThreadStatus : : HIGH_PRIORITY ] ,
kSleeping HighPriThreads ) ;
kSimulated HighPriThreads ) ;
ASSERT_EQ (
ASSERT_EQ (
sleep ing_count[ ThreadStatus : : ThreadType : : ROCKSDB_ LOW_PRIORITY] ,
runn ing_count[ ThreadStatus : : LOW_PRIORITY ] ,
kSleeping LowPriThreads ) ;
kSimulated LowPriThreads ) ;
ASSERT_EQ (
ASSERT_EQ (
sleep ing_count[ ThreadStatus : : ThreadType : : USER_THREAD ] , 0 ) ;
runn ing_count[ ThreadStatus : : USER ] , 0 ) ;
sleeping_task . WakeUp ( ) ;
running_task . FinishAllTasks ( ) ;
sleep ing_task. WaitUntilDone ( ) ;
runn ing_task. WaitUntilDone ( ) ;
// Verify none of the threads are sleep ing
// Verify none of the threads are runn ing
env - > GetThreadList ( & thread_list ) ;
env - > GetThreadList ( & thread_list ) ;
for ( int i = 0 ; i < ThreadStatus : : ThreadType : : TOTAL ; + + i ) {
sleeping_count [ i ] = 0 ;
}
for ( int i = 0 ; i < ThreadStatus : : NUM_THREAD_TYPES ; + + i ) {
running_count [ i ] = 0 ;
}
for ( auto thread_status : thread_list ) {
for ( auto thread_status : thread_list ) {
if ( thread_status . cf_name = = " pikachu " & &
if ( thread_status . cf_name = = " pikachu " & &
thread_status . db_name = = " sleeping " ) {
thread_status . db_name = = " runn ing" ) {
sleep ing_count[ thread_status . thread_type ] + + ;
runn ing_count[ thread_status . thread_type ] + + ;
}
}
}
}
ASSERT_EQ (
ASSERT_EQ (
sleep ing_count[ ThreadStatus : : ThreadType : : ROCKSDB_ HIGH_PRIORITY] , 0 ) ;
runn ing_count[ ThreadStatus : : HIGH_PRIORITY ] , 0 ) ;
ASSERT_EQ (
ASSERT_EQ (
sleep ing_count[ ThreadStatus : : ThreadType : : ROCKSDB_ LOW_PRIORITY] , 0 ) ;
runn ing_count[ ThreadStatus : : LOW_PRIORITY ] , 0 ) ;
ASSERT_EQ (
ASSERT_EQ (
sleeping_count [ ThreadStatus : : ThreadType : : USER_THREAD ] , 0 ) ;
running_count [ ThreadStatus : : USER ] , 0 ) ;
}
namespace {
void UpdateStatusCounts (
const std : : vector < ThreadStatus > & thread_list ,
int operation_counts [ ] , int state_counts [ ] ) {
for ( auto thread_status : thread_list ) {
operation_counts [ thread_status . operation_type ] + + ;
state_counts [ thread_status . state_type ] + + ;
}
}
void VerifyAndResetCounts (
const int correct_counts [ ] , int collected_counts [ ] , int size ) {
for ( int i = 0 ; i < size ; + + i ) {
ASSERT_EQ ( collected_counts [ i ] , correct_counts [ i ] ) ;
collected_counts [ i ] = 0 ;
}
}
void UpdateCount (
int operation_counts [ ] , int from_event , int to_event , int amount ) {
operation_counts [ from_event ] - = amount ;
operation_counts [ to_event ] + = amount ;
}
} // namespace
TEST ( ThreadListTest , SimpleEventTest ) {
Env * env = Env : : Default ( ) ;
// simulated tasks
const int kFlushWriteTasks = 3 ;
SimulatedBackgroundTask flush_write_task (
reinterpret_cast < void * > ( 1234 ) , " running " ,
reinterpret_cast < void * > ( 5678 ) , " pikachu " ,
ThreadStatus : : OP_FLUSH ) ;
const int kCompactionWriteTasks = 4 ;
SimulatedBackgroundTask compaction_write_task (
reinterpret_cast < void * > ( 1234 ) , " running " ,
reinterpret_cast < void * > ( 5678 ) , " pikachu " ,
ThreadStatus : : OP_COMPACTION ) ;
const int kCompactionReadTasks = 5 ;
SimulatedBackgroundTask compaction_read_task (
reinterpret_cast < void * > ( 1234 ) , " running " ,
reinterpret_cast < void * > ( 5678 ) , " pikachu " ,
ThreadStatus : : OP_COMPACTION ) ;
const int kCompactionWaitTasks = 6 ;
SimulatedBackgroundTask compaction_wait_task (
reinterpret_cast < void * > ( 1234 ) , " running " ,
reinterpret_cast < void * > ( 5678 ) , " pikachu " ,
ThreadStatus : : OP_COMPACTION ) ;
// setup right answers
int correct_operation_counts [ ThreadStatus : : NUM_OP_TYPES ] = { 0 } ;
correct_operation_counts [ ThreadStatus : : OP_FLUSH ] =
kFlushWriteTasks ;
correct_operation_counts [ ThreadStatus : : OP_COMPACTION ] =
kCompactionWriteTasks + kCompactionReadTasks + kCompactionWaitTasks ;
env - > SetBackgroundThreads (
correct_operation_counts [ ThreadStatus : : OP_FLUSH ] , Env : : HIGH ) ;
env - > SetBackgroundThreads (
correct_operation_counts [ ThreadStatus : : OP_COMPACTION ] , Env : : LOW ) ;
// schedule the simulated tasks
for ( int t = 0 ; t < kFlushWriteTasks ; + + t ) {
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulatedTask ,
& flush_write_task , Env : : Priority : : HIGH ) ;
}
flush_write_task . WaitUntilScheduled ( kFlushWriteTasks , env ) ;
for ( int t = 0 ; t < kCompactionWriteTasks ; + + t ) {
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulatedTask ,
& compaction_write_task , Env : : Priority : : LOW ) ;
}
compaction_write_task . WaitUntilScheduled ( kCompactionWriteTasks , env ) ;
for ( int t = 0 ; t < kCompactionReadTasks ; + + t ) {
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulatedTask ,
& compaction_read_task , Env : : Priority : : LOW ) ;
}
compaction_read_task . WaitUntilScheduled ( kCompactionReadTasks , env ) ;
for ( int t = 0 ; t < kCompactionWaitTasks ; + + t ) {
env - > Schedule ( & SimulatedBackgroundTask : : DoSimulatedTask ,
& compaction_wait_task , Env : : Priority : : LOW ) ;
}
compaction_wait_task . WaitUntilScheduled ( kCompactionWaitTasks , env ) ;
// verify the thread-status
int operation_counts [ ThreadStatus : : NUM_OP_TYPES ] = { 0 } ;
int state_counts [ ThreadStatus : : NUM_STATE_TYPES ] = { 0 } ;
std : : vector < ThreadStatus > thread_list ;
env - > GetThreadList ( & thread_list ) ;
UpdateStatusCounts ( thread_list , operation_counts , state_counts ) ;
VerifyAndResetCounts ( correct_operation_counts , operation_counts ,
ThreadStatus : : NUM_OP_TYPES ) ;
// terminate compaction-wait tasks and see if the thread-status
// reflects this update
compaction_wait_task . FinishAllTasks ( ) ;
compaction_wait_task . WaitUntilDone ( ) ;
UpdateCount ( correct_operation_counts , ThreadStatus : : OP_COMPACTION ,
ThreadStatus : : OP_UNKNOWN , kCompactionWaitTasks ) ;
env - > GetThreadList ( & thread_list ) ;
UpdateStatusCounts ( thread_list , operation_counts , state_counts ) ;
VerifyAndResetCounts ( correct_operation_counts , operation_counts ,
ThreadStatus : : NUM_OP_TYPES ) ;
// terminate flush-write tasks and see if the thread-status
// reflects this update
flush_write_task . FinishAllTasks ( ) ;
flush_write_task . WaitUntilDone ( ) ;
UpdateCount ( correct_operation_counts , ThreadStatus : : OP_FLUSH ,
ThreadStatus : : OP_UNKNOWN , kFlushWriteTasks ) ;
env - > GetThreadList ( & thread_list ) ;
UpdateStatusCounts ( thread_list , operation_counts , state_counts ) ;
VerifyAndResetCounts ( correct_operation_counts , operation_counts ,
ThreadStatus : : NUM_OP_TYPES ) ;
// terminate compaction-write tasks and see if the thread-status
// reflects this update
compaction_write_task . FinishAllTasks ( ) ;
compaction_write_task . WaitUntilDone ( ) ;
UpdateCount ( correct_operation_counts , ThreadStatus : : OP_COMPACTION ,
ThreadStatus : : OP_UNKNOWN , kCompactionWriteTasks ) ;
env - > GetThreadList ( & thread_list ) ;
UpdateStatusCounts ( thread_list , operation_counts , state_counts ) ;
VerifyAndResetCounts ( correct_operation_counts , operation_counts ,
ThreadStatus : : NUM_OP_TYPES ) ;
// terminate compaction-write tasks and see if the thread-status
// reflects this update
compaction_read_task . FinishAllTasks ( ) ;
compaction_read_task . WaitUntilDone ( ) ;
UpdateCount ( correct_operation_counts , ThreadStatus : : OP_COMPACTION ,
ThreadStatus : : OP_UNKNOWN , kCompactionReadTasks ) ;
env - > GetThreadList ( & thread_list ) ;
UpdateStatusCounts ( thread_list , operation_counts , state_counts ) ;
VerifyAndResetCounts ( correct_operation_counts , operation_counts ,
ThreadStatus : : NUM_OP_TYPES ) ;
}
}
} // namespace rocksdb
} // namespace rocksdb