blob: 46ade7ad6233c43bed7d43dbdf670a7bca9d009f [file] [log] [blame]
/*
* Copyright 2021 The Android Open Source Project
*
* 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/bind.h>
#include <base/location.h>
#include <gtest/gtest.h>
#include "bta/dm/bta_dm_int.h"
#include "bta/hf_client/bta_hf_client_int.h"
#include "bta/include/bta_dm_api.h"
#include "bta/include/bta_hf_client_api.h"
#include "btif/include/stack_manager.h"
#include "common/message_loop_thread.h"
#include "test/mock/mock_osi_alarm.h"
#include "test/mock/mock_stack_acl.h"
std::map<std::string, int> mock_function_count_map;
extern struct btm_client_interface_s btm_client_interface;
void LogMsg(uint32_t trace_set_mask, const char* fmt_str, ...) {}
namespace base {
class MessageLoop;
} // namespace base
namespace {
constexpr uint8_t kUnusedTimer = BTA_ID_MAX;
} // namespace
struct alarm_t {
alarm_t(const char* name){};
int any_value;
};
class BtaDmTest : public testing::Test {
protected:
void SetUp() override {
mock_function_count_map.clear();
test::mock::osi_alarm::alarm_new.body = [](const char* name) -> alarm_t* {
return new alarm_t(name);
};
test::mock::osi_alarm::alarm_free.body = [](alarm_t* alarm) {
delete alarm;
};
bta_dm_init_cb();
for (int i = 0; i < BTA_DM_NUM_PM_TIMER; i++) {
for (int j = 0; j < BTA_DM_PM_MODE_TIMER_MAX; j++) {
bta_dm_cb.pm_timer[i].srvc_id[j] = kUnusedTimer;
}
}
}
void TearDown() override {
bta_dm_deinit_cb();
test::mock::osi_alarm::alarm_new = {};
test::mock::osi_alarm::alarm_free = {};
}
};
TEST_F(BtaDmTest, nop) {
bool status = true;
ASSERT_EQ(true, status);
}
TEST_F(BtaDmTest, disable_no_acl_links) {
bta_dm_cb.disabling = true;
alarm_callback_t alarm_callback;
void* alarm_data{nullptr};
test::mock::osi_alarm::alarm_set_on_mloop.body =
[&alarm_callback, &alarm_data](alarm_t* alarm, uint64_t interval_ms,
alarm_callback_t cb, void* data) {
ASSERT_TRUE(alarm != nullptr);
alarm_callback = cb;
alarm_data = data;
};
bta_dm_disable(); // Waiting for all ACL connections to drain
ASSERT_EQ(0, mock_function_count_map["btm_remove_acl"]);
ASSERT_EQ(1, mock_function_count_map["alarm_set_on_mloop"]);
// Execute timer callback
alarm_callback(alarm_data);
ASSERT_EQ(1, mock_function_count_map["alarm_set_on_mloop"]);
ASSERT_EQ(0, mock_function_count_map["BTIF_dm_disable"]);
ASSERT_EQ(1, mock_function_count_map["future_ready"]);
ASSERT_TRUE(!bta_dm_cb.disabling);
test::mock::osi_alarm::alarm_set_on_mloop = {};
}
TEST_F(BtaDmTest, disable_first_pass_with_acl_links) {
uint16_t links_up = 1;
test::mock::stack_acl::BTM_GetNumAclLinks.body = [&links_up]() {
return links_up;
};
bta_dm_cb.disabling = true;
// ACL link is open
bta_dm_cb.device_list.count = 1;
alarm_callback_t alarm_callback;
void* alarm_data{nullptr};
test::mock::osi_alarm::alarm_set_on_mloop.body =
[&alarm_callback, &alarm_data](alarm_t* alarm, uint64_t interval_ms,
alarm_callback_t cb, void* data) {
ASSERT_TRUE(alarm != nullptr);
alarm_callback = cb;
alarm_data = data;
};
bta_dm_disable(); // Waiting for all ACL connections to drain
ASSERT_EQ(1, mock_function_count_map["alarm_set_on_mloop"]);
ASSERT_EQ(0, mock_function_count_map["BTIF_dm_disable"]);
links_up = 0;
// First disable pass
alarm_callback(alarm_data);
ASSERT_EQ(1, mock_function_count_map["alarm_set_on_mloop"]);
ASSERT_EQ(1, mock_function_count_map["BTIF_dm_disable"]);
ASSERT_TRUE(!bta_dm_cb.disabling);
test::mock::stack_acl::BTM_GetNumAclLinks = {};
test::mock::osi_alarm::alarm_set_on_mloop = {};
}
TEST_F(BtaDmTest, disable_second_pass_with_acl_links) {
uint16_t links_up = 1;
test::mock::stack_acl::BTM_GetNumAclLinks.body = [&links_up]() {
return links_up;
};
bta_dm_cb.disabling = true;
// ACL link is open
bta_dm_cb.device_list.count = 1;
alarm_callback_t alarm_callback;
void* alarm_data{nullptr};
test::mock::osi_alarm::alarm_set_on_mloop.body =
[&alarm_callback, &alarm_data](alarm_t* alarm, uint64_t interval_ms,
alarm_callback_t cb, void* data) {
ASSERT_TRUE(alarm != nullptr);
alarm_callback = cb;
alarm_data = data;
};
bta_dm_disable(); // Waiting for all ACL connections to drain
ASSERT_EQ(1, mock_function_count_map["alarm_set_on_mloop"]);
ASSERT_EQ(0, mock_function_count_map["BTIF_dm_disable"]);
// First disable pass
alarm_callback(alarm_data);
ASSERT_EQ(2, mock_function_count_map["alarm_set_on_mloop"]);
ASSERT_EQ(0, mock_function_count_map["BTIF_dm_disable"]);
ASSERT_EQ(1, mock_function_count_map["btm_remove_acl"]);
// Second disable pass
alarm_callback(alarm_data);
ASSERT_EQ(1, mock_function_count_map["BTIF_dm_disable"]);
ASSERT_TRUE(!bta_dm_cb.disabling);
test::mock::stack_acl::BTM_GetNumAclLinks = {};
test::mock::osi_alarm::alarm_set_on_mloop = {};
}