/****************************************************************************** * * Copyright 2014 Google, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ******************************************************************************/ #include <base/message_loop/message_loop.h> #include <base/run_loop.h> #include <gtest/gtest.h> #include "AlarmTestHarness.h" #include "common/message_loop_thread.h" #include "osi/include/alarm.h" #include "osi/include/fixed_queue.h" #include "osi/include/osi.h" #include "osi/include/semaphore.h" using base::Closure; using base::TimeDelta; using bluetooth::common::MessageLoopThread; static semaphore_t* semaphore; static int cb_counter; static int cb_misordered_counter; static const uint64_t EPSILON_MS = 50; static void msleep(uint64_t ms) { usleep(ms * 1000); } static base::MessageLoop* message_loop_; base::MessageLoop* get_main_message_loop() { return message_loop_; } class AlarmTest : public AlarmTestHarness { protected: virtual void SetUp() { AlarmTestHarness::SetUp(); cb_counter = 0; cb_misordered_counter = 0; semaphore = semaphore_new(0); } virtual void TearDown() { semaphore_free(semaphore); AlarmTestHarness::TearDown(); } }; static void cb(UNUSED_ATTR void* data) { ++cb_counter; semaphore_post(semaphore); } static void ordered_cb(void* data) { int i = PTR_TO_INT(data); if (i != cb_counter) cb_misordered_counter++; ++cb_counter; semaphore_post(semaphore); } TEST_F(AlarmTest, test_new_free_simple) { alarm_t* alarm = alarm_new("alarm_test.test_new_free_simple"); ASSERT_TRUE(alarm != NULL); alarm_free(alarm); } TEST_F(AlarmTest, test_free_null) { alarm_free(NULL); } TEST_F(AlarmTest, test_simple_cancel) { alarm_t* alarm = alarm_new("alarm_test.test_simple_cancel"); alarm_cancel(alarm); alarm_free(alarm); } TEST_F(AlarmTest, test_cancel) { alarm_t* alarm = alarm_new("alarm_test.test_cancel"); alarm_set(alarm, 10, cb, NULL); alarm_cancel(alarm); msleep(10 + EPSILON_MS); EXPECT_EQ(cb_counter, 0); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } TEST_F(AlarmTest, test_cancel_idempotent) { alarm_t* alarm = alarm_new("alarm_test.test_cancel_idempotent"); alarm_set(alarm, 10, cb, NULL); alarm_cancel(alarm); alarm_cancel(alarm); alarm_cancel(alarm); alarm_free(alarm); } TEST_F(AlarmTest, test_set_short) { alarm_t* alarm = alarm_new("alarm_test.test_set_short"); alarm_set(alarm, 10, cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_TRUE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 1); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } TEST_F(AlarmTest, test_set_short_periodic) { alarm_t* alarm = alarm_new_periodic("alarm_test.test_set_short_periodic"); alarm_set(alarm, 10, cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_TRUE(WakeLockHeld()); for (int i = 1; i <= 10; i++) { semaphore_wait(semaphore); EXPECT_GE(cb_counter, i); EXPECT_TRUE(WakeLockHeld()); } alarm_cancel(alarm); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } TEST_F(AlarmTest, test_set_zero_periodic) { alarm_t* alarm = alarm_new_periodic("alarm_test.test_set_zero_periodic"); alarm_set(alarm, 0, cb, NULL); EXPECT_TRUE(WakeLockHeld()); for (int i = 1; i <= 10; i++) { semaphore_wait(semaphore); EXPECT_GE(cb_counter, i); EXPECT_TRUE(WakeLockHeld()); } alarm_cancel(alarm); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } TEST_F(AlarmTest, test_set_long) { alarm_t* alarm = alarm_new("alarm_test.test_set_long"); alarm_set(alarm, TIMER_INTERVAL_FOR_WAKELOCK_IN_MS + EPSILON_MS, cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_FALSE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 1); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } TEST_F(AlarmTest, test_set_short_short) { alarm_t* alarm[2] = {alarm_new("alarm_test.test_set_short_short_0"), alarm_new("alarm_test.test_set_short_short_1")}; alarm_set(alarm[0], 10, cb, NULL); alarm_set(alarm[1], 20, cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_TRUE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 1); EXPECT_TRUE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 2); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm[0]); alarm_free(alarm[1]); } TEST_F(AlarmTest, test_set_short_long) { alarm_t* alarm[2] = {alarm_new("alarm_test.test_set_short_long_0"), alarm_new("alarm_test.test_set_short_long_1")}; alarm_set(alarm[0], 10, cb, NULL); alarm_set(alarm[1], 10 + TIMER_INTERVAL_FOR_WAKELOCK_IN_MS + EPSILON_MS, cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_TRUE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 1); EXPECT_FALSE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 2); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm[0]); alarm_free(alarm[1]); } TEST_F(AlarmTest, test_set_long_long) { alarm_t* alarm[2] = {alarm_new("alarm_test.test_set_long_long_0"), alarm_new("alarm_test.test_set_long_long_1")}; alarm_set(alarm[0], TIMER_INTERVAL_FOR_WAKELOCK_IN_MS + EPSILON_MS, cb, NULL); alarm_set(alarm[1], 2 * (TIMER_INTERVAL_FOR_WAKELOCK_IN_MS + EPSILON_MS), cb, NULL); EXPECT_EQ(cb_counter, 0); EXPECT_FALSE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 1); EXPECT_FALSE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_EQ(cb_counter, 2); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm[0]); alarm_free(alarm[1]); } TEST_F(AlarmTest, test_is_scheduled) { alarm_t* alarm = alarm_new("alarm_test.test_is_scheduled"); EXPECT_FALSE(alarm_is_scheduled((alarm_t*)NULL)); EXPECT_FALSE(alarm_is_scheduled(alarm)); alarm_set(alarm, TIMER_INTERVAL_FOR_WAKELOCK_IN_MS + EPSILON_MS, cb, NULL); EXPECT_TRUE(alarm_is_scheduled(alarm)); EXPECT_EQ(cb_counter, 0); EXPECT_FALSE(WakeLockHeld()); semaphore_wait(semaphore); EXPECT_FALSE(alarm_is_scheduled(alarm)); EXPECT_EQ(cb_counter, 1); EXPECT_FALSE(WakeLockHeld()); alarm_free(alarm); } // Test whether the callbacks are invoked in the expected order TEST_F(AlarmTest, test_callback_ordering) { alarm_t* alarms[100]; for (int i = 0; i < 100; i++) { const std::string alarm_name = "alarm_test.test_callback_ordering[" + std::to_string(i) + "]"; alarms[i] = alarm_new(alarm_name.c_str()); } for (int i = 0; i < 100; i++) { alarm_set(alarms[i], 100, ordered_cb, INT_TO_PTR(i)); } for (int i = 1; i <= 100; i++) { semaphore_wait(semaphore); EXPECT_GE(cb_counter, i); } EXPECT_EQ(cb_counter, 100); EXPECT_EQ(cb_misordered_counter, 0); for (int i = 0; i < 100; i++) alarm_free(alarms[i]); EXPECT_FALSE(WakeLockHeld()); } // Test whether the callbacks are involed in the expected order on a // message loop. TEST_F(AlarmTest, test_callback_ordering_on_mloop) { alarm_t* alarms[100]; // Initialize MesageLoop, and wait till it's initialized. MessageLoopThread message_loop_thread("btu message loop"); message_loop_thread.StartUp(); if (!message_loop_thread.IsRunning()) { FAIL() << "unable to create btu message loop thread."; } message_loop_ = message_loop_thread.message_loop(); for (int i = 0; i < 100; i++) { const std::string alarm_name = "alarm_test.test_callback_ordering_on_mloop[" + std::to_string(i) + "]"; alarms[i] = alarm_new(alarm_name.c_str()); } for (int i = 0; i < 100; i++) { alarm_set_on_mloop(alarms[i], 100, ordered_cb, INT_TO_PTR(i)); } for (int i = 1; i <= 100; i++) { semaphore_wait(semaphore); EXPECT_GE(cb_counter, i); } EXPECT_EQ(cb_counter, 100); EXPECT_EQ(cb_misordered_counter, 0); for (int i = 0; i < 100; i++) alarm_free(alarms[i]); message_loop_thread.ShutDown(); EXPECT_FALSE(WakeLockHeld()); } // Try to catch any race conditions between the timer callback and |alarm_free|. TEST_F(AlarmTest, test_callback_free_race) { for (int i = 0; i < 1000; ++i) { const std::string alarm_name = "alarm_test.test_callback_free_race[" + std::to_string(i) + "]"; alarm_t* alarm = alarm_new(alarm_name.c_str()); alarm_set(alarm, 0, cb, NULL); alarm_free(alarm); } alarm_cleanup(); } static void remove_cb(void* data) { alarm_free((alarm_t*)data); semaphore_post(semaphore); } TEST_F(AlarmTest, test_delete_during_callback) { for (int i = 0; i < 1000; ++i) { alarm_t* alarm = alarm_new("alarm_test.test_delete_during_callback"); alarm_set(alarm, 0, remove_cb, alarm); semaphore_wait(semaphore); } alarm_cleanup(); }