@ -15,272 +15,187 @@ class TimerTest : public testing::Test { 
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					 protected :   protected :   
			
		
	
		
		
			
				
					
					  std : : unique_ptr < MockTimeEnv >  mock_env_ ;    std : : unique_ptr < MockTimeEnv >  mock_env_ ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					# if defined(OS_MACOSX) && !defined(NDEBUG)  
			
		
	
		
		
			
				
					
					  // On some platforms (MacOS) pthread_cond_timedwait does not appear
   
			
		
	
		
		
			
				
					
					  // to release the lock for other threads to operate if the deadline time
   
			
		
	
		
		
			
				
					
					  // is already passed. This is a problem for tests in general because
   
			
		
	
		
		
			
				
					
					  // TimedWait calls are a bad abstraction: the deadline parameter is
   
			
		
	
		
		
			
				
					
					  // usually computed from Env time, but is interpreted in real clock time.
   
			
		
	
		
		
			
				
					
					  // Since this test doesn't even pretend to use clock times, we have
   
			
		
	
		
		
			
				
					
					  // to mock TimedWait to ensure it yields.
   
			
		
	
		
		
			
				
					
					  void  SetUp ( )  override  {   
			
		
	
		
		
			
				
					
					    ROCKSDB_NAMESPACE : : SyncPoint : : GetInstance ( ) - > DisableProcessing ( ) ;   
			
		
	
		
		
			
				
					
					    ROCKSDB_NAMESPACE : : SyncPoint : : GetInstance ( ) - > ClearAllCallBacks ( ) ;   
			
		
	
		
		
			
				
					
					    ROCKSDB_NAMESPACE : : SyncPoint : : GetInstance ( ) - > SetCallBack (   
			
		
	
		
		
			
				
					
					        " InstrumentedCondVar::TimedWaitInternal " ,  [ & ] ( void *  arg )  {   
			
		
	
		
		
			
				
					
					          uint64_t *  time_us  =  reinterpret_cast < uint64_t * > ( arg ) ;   
			
		
	
		
		
			
				
					
					          if  ( * time_us  <  mock_env_ - > RealNowMicros ( ) )  {   
			
		
	
		
		
			
				
					
					            * time_us  =  mock_env_ - > RealNowMicros ( )  +  1000 ;   
			
		
	
		
		
			
				
					
					          }   
			
		
	
		
		
			
				
					
					        } ) ;   
			
		
	
		
		
			
				
					
					    ROCKSDB_NAMESPACE : : SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
			
				
					
					# endif   // OS_MACOSX && !NDEBUG
  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  const  uint64_t  kSecond  =  1000000 ;   // 1sec = 1000000us
    const  uint64_t  kSecond  =  1000000 ;   // 1sec = 1000000us
   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  void  SetUp ( )  override  {  mock_env_ - > InstallTimedWaitFixCallback ( ) ;  }   
			
		
	
		
		
			
				
					
					} ; } ;  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  SingleScheduleOnceTest )  { TEST_F ( TimerTest ,  SingleScheduleOnceTest )  {  
			
		
	
		
		
			
				
					
					  const  int  kIterations  =  1 ;    const  int  kInitDelaySec  =  1 ;   
			
				
				
			
		
	
		
		
			
				
					
					  uint64_t  time_counter  =  0 ;    int  mock_time_sec  =  0 ;   
			
				
				
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
				
				
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex ;   
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv ( & mutex ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  count  =  0 ;    int  count  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count + + ;  } ,  " fn_sch_test " ,  kInitDelaySec  *  kSecond ,  0 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {   
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex ) ;   
			
		
	
		
		
			
				
					
					        count + + ;   
			
		
	
		
		
			
				
					
					        if  ( count  > =  kIterations )  {   
			
		
	
		
		
			
				
					
					          test_cv . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test " ,  1  *  kSecond ,  0 ) ;   
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count ) ;   
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    // Wait for execution to finish
   
			
		
	
		
		
			
				
					
					  {    mock_time_sec  + =  kInitDelaySec ;   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex ) ;    timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    while ( count  <  kIterations )  {    ASSERT_EQ ( 1 ,  count ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;   
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;   
			
		
	
		
		
			
				
					
					      test_cv . TimedWait ( time_counter ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  count ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  MultipleScheduleOnceTest )  { TEST_F ( TimerTest ,  MultipleScheduleOnceTest )  {  
			
		
	
		
		
			
				
					
					  const  int  kIterations  =  1 ;    const  int  kInitDelay1Sec  =  1 ;   
			
				
				
			
		
	
		
		
			
				
					
					  uint64_t  time_counter  =  0 ;    const  int  kInitDelay2Sec  =  3 ;   
			
				
				
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    int  mock_time_sec  =  0 ;   
			
				
				
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex1 ;    mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv1 ( & mutex1 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  count1  =  0 ;    int  count1  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count1 + + ;  } ,  " fn_sch_test1 " ,  kInitDelay1Sec  *  kSecond ,  0 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {   
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex1 ) ;   
			
		
	
		
		
			
				
					
					        count1 + + ;   
			
		
	
		
		
			
				
					
					        if  ( count1  > =  kIterations )  {   
			
		
	
		
		
			
				
					
					          test_cv1 . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test1 " ,  1  *  kSecond ,  0 ) ;   
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex2 ;   
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv2 ( & mutex2 ) ;   
			
		
	
		
		
			
				
					
					  int  count2  =  0 ;    int  count2  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count2 + + ;  } ,  " fn_sch_test2 " ,  kInitDelay2Sec  *  kSecond ,  0 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {   
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex2 ) ;   
			
		
	
		
		
			
				
					
					        count2  + =  5 ;   
			
		
	
		
		
			
				
					
					        if  ( count2  > =  kIterations )  {   
			
		
	
		
		
			
				
					
					          test_cv2 . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test2 " ,  3  *  kSecond ,  0 ) ;   
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count1 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count2 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    mock_time_sec  =  kInitDelay1Sec ;   
			
				
				
			
		
	
		
		
			
				
					
					  {    timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex1 ) ;   
			
		
	
		
		
			
				
					
					    while  ( count1  <  kIterations )  {   
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;   
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;   
			
		
	
		
		
			
				
					
					      test_cv1 . TimedWait ( time_counter ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    ASSERT_EQ ( 1 ,  count1 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  {    ASSERT_EQ ( 0 ,  count2 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex2 ) ;   
			
		
	
		
		
			
				
					
					    while ( count2  <  kIterations )  {   
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;   
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;   
			
		
	
		
		
			
				
					
					      test_cv2 . TimedWait ( time_counter ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    mock_time_sec  =  kInitDelay2Sec ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  count1 ) ;    ASSERT_EQ ( 1 ,  count1 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 5 ,  count2 ) ;    ASSERT_EQ ( 1 ,  count2 ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  SingleScheduleRepeatedlyTest )  { TEST_F ( TimerTest ,  SingleScheduleRepeatedlyTest )  {  
			
		
	
		
		
			
				
					
					  const  int  kIterations  =  5 ;    const  int  kIterations  =  5 ;   
			
		
	
		
		
			
				
					
					  uint64_t  time_counter  =  0 ;    const  int  kInitDelaySec  =  1 ;   
			
				
				
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    const  int  kRepeatSec  =  1 ;   
			
				
				
			
		
	
		
		
			
				
					
					
  int  mock_time_sec  =  0 ;   
			
				
				
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex ;    mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv ( & mutex ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					  int  count  =  0 ;    int  count  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count + + ;  } ,  " fn_sch_test " ,  kInitDelaySec  *  kSecond ,   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {              kRepeatSec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex ) ;   
			
		
	
		
		
			
				
					
					        count + + ;   
			
		
	
		
		
			
				
					
					        if  ( count  > =  kIterations )  {   
			
		
	
		
		
			
				
					
					          test_cv . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test " ,  1  *  kSecond ,  1  *  kSecond ) ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mock_time_sec  + =  kInitDelaySec ;   
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  count ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    // Wait for execution to finish
   
			
		
	
		
		
			
				
					
					  {    for  ( int  i  =  1 ;  i  <  kIterations ;  i + + )  {   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex ) ;      mock_time_sec  + =  kRepeatSec ;   
			
				
				
			
		
	
		
		
			
				
					
					    while ( count  <  kIterations )  {      timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;   
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;   
			
		
	
		
		
			
				
					
					      test_cv . TimedWait ( time_counter ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( kIterations ,  count ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 5 ,  count ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  MultipleScheduleRepeatedlyTest )  { TEST_F ( TimerTest ,  MultipleScheduleRepeatedlyTest )  {  
			
		
	
		
		
			
				
					
					  uint64_t  time_counter  =  0 ;    const  int  kInitDelay1Sec  =  0 ;   
			
				
				
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    const  int  kInitDelay2Sec  =  1 ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  const  int  kInitDelay3Sec  =  0 ;   
			
		
	
		
		
			
				
					
					  const  int  kRepeatSec  =  2 ;   
			
		
	
		
		
			
				
					
					  const  int  kLargeRepeatSec  =  100 ;   
			
		
	
		
		
			
				
					
					  const  int  kIterations  =  5 ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  mock_time_sec  =  0 ;   
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex1 ;   
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv1 ( & mutex1 ) ;   
			
		
	
		
		
			
				
					
					  const  int  kIterations1  =  5 ;   
			
		
	
		
		
			
				
					
					  int  count1  =  0 ;    int  count1  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count1 + + ;  } ,  " fn_sch_test1 " ,  kInitDelay1Sec  *  kSecond ,   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {              kRepeatSec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex1 ) ;   
			
		
	
		
		
			
				
					
					        count1 + + ;   
			
		
	
		
		
			
				
					
					        if  ( count1  > =  kIterations1 )  {   
			
		
	
		
		
			
				
					
					          test_cv1 . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test1 " ,  0 ,  2  *  kSecond ) ;   
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex2 ;   
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv2 ( & mutex2 ) ;   
			
		
	
		
		
			
				
					
					  const  int  kIterations2  =  5 ;   
			
		
	
		
		
			
				
					
					  int  count2  =  0 ;    int  count2  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count2 + + ;  } ,  " fn_sch_test2 " ,  kInitDelay2Sec  *  kSecond ,   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {              kRepeatSec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex2 ) ;  
 
			
				
				
			
		
	
		
		
			
				
					
					        count2 + + ;    // Add a function with relatively large repeat interval
   
			
				
				
			
		
	
		
		
			
				
					
					        if  ( count2  > =  kIterations2 )  {    int  count3  =  0 ;   
			
				
				
			
		
	
		
		
			
				
					
					          test_cv2 . SignalAll ( ) ;    timer . Add ( [ & ]  {  count3 + + ;  } ,  " fn_sch_test3 " ,  kInitDelay3Sec  *  kSecond ,   
			
				
				
			
		
	
		
		
			
				
					
					        }              kLargeRepeatSec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test2 " ,  1  *  kSecond ,  2  *  kSecond ) ;   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count2 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  count3 ) ;   
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    // Wait for execution to finish
   
			
		
	
		
		
			
				
					
					  {    for  ( ;  count1  <  kIterations ;  mock_time_sec + + )  {   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex1 ) ;      timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    while ( count1  <  kIterations1 )  {      ASSERT_EQ ( ( mock_time_sec  +  2 )  /  kRepeatSec ,  count1 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond  ;      ASSERT_EQ ( ( mock_time_sec  +  1 )  /  kRepeatSec ,  count2 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;  
  
			
				
				
			
		
	
		
		
			
				
					
					      test_cv1 . TimedWait ( time_counter ) ;       // large interval function should only run once (the first one).
   
			
				
				
			
		
	
		
		
			
				
					
					    }      ASSERT_EQ ( 1 ,  count3 ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  timer . Cancel ( " fn_sch_test1 " ) ;    timer . Cancel ( " fn_sch_test1 " ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // Wait for execution to finish
    // Wait for execution to finish
   
			
		
	
		
		
			
				
					
					  {    mock_time_sec + + ;   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex2 ) ;    timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    while ( count2  <  kIterations2 )  {    ASSERT_EQ ( kIterations ,  count1 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;    ASSERT_EQ ( kIterations ,  count2 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;    ASSERT_EQ ( 1 ,  count3 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					      test_cv2 . TimedWait ( time_counter ) ;   
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
			
				
					
					  }   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  timer . Cancel ( " fn_sch_test2 " ) ;    timer . Cancel ( " fn_sch_test2 " ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    ASSERT_EQ ( kIterations ,  count1 ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  ASSERT_EQ ( kIterations ,  count2 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( count1 ,  5 ) ;    // execute the long interval one
   
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( count2 ,  5 ) ;    mock_time_sec  =  kLargeRepeatSec ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 2 ,  count3 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  AddAfterStartTest )  { TEST_F ( TimerTest ,  AddAfterStartTest )  {  
			
		
	
		
		
			
				
					
					  const  int  kIterations  =  5 ;    const  int  kIterations  =  5 ;   
			
		
	
		
		
			
				
					
					  InstrumentedMutex  mutex ;    const  int  kInitDelaySec  =  1 ;   
			
				
				
			
		
	
		
		
			
				
					
					  InstrumentedCondVar  test_cv ( & mutex ) ;    const  int  kRepeatSec  =  1 ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // wait timer to run and then add a new job
    // wait timer to run and then add a new job
   
			
		
	
		
		
			
				
					
					  SyncPoint : : GetInstance ( ) - > LoadDependency (    SyncPoint : : GetInstance ( ) - > LoadDependency (   
			
		
	
		
		
			
				
					
					      { { " Timer::Run::Waiting " ,  " TimerTest:AddAfterStartTest:1 " } } ) ;        { { " Timer::Run::Waiting " ,  " TimerTest:AddAfterStartTest:1 " } } ) ;   
			
		
	
		
		
			
				
					
					  SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;    SyncPoint : : GetInstance ( ) - > EnableProcessing ( ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    int  mock_time_sec  =  0 ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  TEST_SYNC_POINT ( " TimerTest:AddAfterStartTest:1 " ) ;    TEST_SYNC_POINT ( " TimerTest:AddAfterStartTest:1 " ) ;   
			
		
	
		
		
			
				
					
					  int  count  =  0 ;    int  count  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add (    timer . Add ( [ & ]  {  count + + ;  } ,  " fn_sch_test " ,  kInitDelaySec  *  kSecond ,   
			
				
				
			
		
	
		
		
			
				
					
					      [ & ]  {              kRepeatSec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        InstrumentedMutexLock  l ( & mutex ) ;    ASSERT_EQ ( 0 ,  count ) ;   
			
				
				
			
		
	
		
		
			
				
					
					        count + + ;   
			
		
	
		
		
			
				
					
					        if  ( count  > =  kIterations )  {   
			
		
	
		
		
			
				
					
					          test_cv . SignalAll ( ) ;   
			
		
	
		
		
			
				
					
					        }   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " fn_sch_test " ,  1  *  kSecond ,  1  *  kSecond ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  // Wait for execution to finish
    // Wait for execution to finish
   
			
		
	
		
		
			
				
					
					  uint64_t  time_counter  =  0 ;    mock_time_sec  + =  kInitDelaySec ;   
			
				
				
			
		
	
		
		
			
				
					
					  {    timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    InstrumentedMutexLock  l ( & mutex ) ;    ASSERT_EQ ( 1 ,  count ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    while  ( count  <  kIterations )  {  
 
			
				
				
			
		
	
		
		
			
				
					
					      time_counter  + =  kSecond ;    for  ( int  i  =  1 ;  i  <  kIterations ;  i + + )  {   
			
				
				
			
		
	
		
		
			
				
					
					      mock_env_ - > set_current_time ( time_counter ) ;      mock_time_sec  + =  kRepeatSec ;   
			
				
				
			
		
	
		
		
			
				
					
					      test_cv . TimedWait ( time_counter ) ;      timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
				
				
			
		
	
		
		
			
				
					
					    }   
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  }    }   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( kIterations ,  count ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( kIterations ,  count ) ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  CancelRunningTask )  { TEST_F ( TimerTest ,  CancelRunningTask )  {  
			
		
	
	
		
		
			
				
					
						
							
								 
						
						
							
								 
						
						
					 
					@ -356,35 +271,86 @@ TEST_F(TimerTest, ShutdownRunningTask) { 
			
		
	
		
		
			
				
					
					  delete  value ;    delete  value ;   
			
		
	
		
		
			
				
					
					} }  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  AddSameFuncNameTest )  { TEST_F ( TimerTest ,  AddSameFuncName )  {  
			
				
				
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( 0 ) ;    const  int  kInitDelaySec  =  1 ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					  const  int  kRepeat1Sec  =  5 ;   
			
		
	
		
		
			
				
					
					  const  int  kRepeat2Sec  =  4 ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  mock_time_sec  =  0 ;   
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;    Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Start ( ) ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  func_counter1  =  0 ;    int  func_counter1  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add ( [ & ]  {  func_counter1 + + ;  } ,  " duplicated_func " ,  1  *  kSecond ,     timer . Add ( [ & ]  {  func_counter1 + + ;  } ,  " duplicated_func " ,   
			
				
				
			
		
	
		
		
			
				
					
					            5 *  kSecond ) ;              kInitDelaySec  *  kSecond ,  kRepeat1Sec *  kSecond ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  func2_counter  =  0 ;    int  func2_counter  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add ( [ & ]  {  func2_counter + + ;  } ,  " func2 " ,  1  *  kSecond ,  4  *  kSecond ) ;    timer . Add ( [ & ]  {  func2_counter + + ;  } ,  " func2 " ,  kInitDelaySec  *  kSecond ,   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					            kRepeat2Sec  *  kSecond ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // New function with the same name should override the existing one
    // New function with the same name should override the existing one
   
			
		
	
		
		
			
				
					
					  int  func_counter2  =  0 ;    int  func_counter2  =  0 ;   
			
		
	
		
		
			
				
					
					  timer . Add ( [ & ]  {  func_counter2 + + ;  } ,  " duplicated_func " ,  1  *  kSecond ,    timer . Add ( [ & ]  {  func_counter2 + + ;  } ,  " duplicated_func " ,   
			
				
				
			
		
	
		
		
			
				
					
					            5  *  kSecond ) ;              kInitDelaySec  *  kSecond ,  kRepeat1Sec  *  kSecond ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  func_counter1 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  func2_counter ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  func_counter2 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mock_time_sec  + =  kInitDelaySec ;   
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  func_counter1 ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  func2_counter ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  func_counter2 ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( 1 ) ;  } ) ;    mock_time_sec  + =  kRepeat1Sec ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func_counter1 ,  0 ) ;    ASSERT_EQ ( 0 ,  func_counter1 ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func2_counter ,  1 ) ;    ASSERT_EQ ( 2 ,  func2_counter ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func_counter2 ,  1 ) ;    ASSERT_EQ ( 2 ,  func_counter2 ) ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( 6 ) ;  } ) ;    ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					}  
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					TEST_F ( TimerTest ,  RepeatIntervalWithFuncRunningTime )  {  
			
		
	
		
		
			
				
					
					  const  int  kInitDelaySec  =  1 ;   
			
		
	
		
		
			
				
					
					  const  int  kRepeatSec  =  5 ;   
			
		
	
		
		
			
				
					
					  const  int  kFuncRunningTimeSec  =  1 ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  int  mock_time_sec  =  0 ;   
			
		
	
		
		
			
				
					
					  mock_env_ - > set_current_time ( mock_time_sec ) ;   
			
		
	
		
		
			
				
					
					  Timer  timer ( mock_env_ . get ( ) ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func_counter1 ,  0 ) ;    ASSERT_TRUE ( timer . Start ( ) ) ;   
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func2_counter ,  2 ) ;  
 
			
				
				
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( func_counter2 ,  2 ) ;    int  func_counter  =  0 ;   
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					  timer . Add (   
			
		
	
		
		
			
				
					
					      [ & ]  {   
			
		
	
		
		
			
				
					
					        mock_env_ - > set_current_time ( mock_time_sec  +  kFuncRunningTimeSec ) ;   
			
		
	
		
		
			
				
					
					        func_counter + + ;   
			
		
	
		
		
			
				
					
					      } ,   
			
		
	
		
		
			
				
					
					      " func " ,  kInitDelaySec  *  kSecond ,  kRepeatSec  *  kSecond ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 0 ,  func_counter ) ;   
			
		
	
		
		
			
				
					
					  mock_time_sec  + =  kInitDelaySec ;   
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  func_counter ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  // After repeat interval time, the function is not executed, as running
   
			
		
	
		
		
			
				
					
					  // the function takes some time (`kFuncRunningTimeSec`). The repeat interval
   
			
		
	
		
		
			
				
					
					  // is the time between ending time of the last call and starting time of the
   
			
		
	
		
		
			
				
					
					  // next call.
   
			
		
	
		
		
			
				
					
					  mock_time_sec  + =  kRepeatSec ;   
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 1 ,  func_counter ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  mock_time_sec  + =  kFuncRunningTimeSec ;   
			
		
	
		
		
			
				
					
					  timer . TEST_WaitForRun ( [ & ]  {  mock_env_ - > set_current_time ( mock_time_sec ) ;  } ) ;   
			
		
	
		
		
			
				
					
					  ASSERT_EQ ( 2 ,  func_counter ) ;   
			
		
	
		
		
			
				
					
					
 
			
		
	
		
		
			
				
					
					  ASSERT_TRUE ( timer . Shutdown ( ) ) ;    ASSERT_TRUE ( timer . Shutdown ( ) ) ;   
			
		
	
		
		
			
				
					
					} }