/* * Copyright 2019 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 "stack_manager.h" #include #include #include #include #include "common/bind.h" #include "module.h" #include "os/handler.h" #include "os/log.h" #include "os/system_properties.h" #include "os/thread.h" #include "os/wakelock_manager.h" using ::bluetooth::os::Handler; using ::bluetooth::os::Thread; using ::bluetooth::os::WakelockManager; namespace bluetooth { void StackManager::StartUp(ModuleList* modules, Thread* stack_thread) { management_thread_ = new Thread("management_thread", Thread::Priority::NORMAL); handler_ = new Handler(management_thread_); WakelockManager::Get().Acquire(); std::promise promise; auto future = promise.get_future(); handler_->Post(common::BindOnce(&StackManager::handle_start_up, common::Unretained(this), modules, stack_thread, std::move(promise))); auto init_status = future.wait_for(std::chrono::milliseconds( get_gd_stack_timeout_ms(/* is_start = */ true))); WakelockManager::Get().Release(); log::info("init_status == {}", int(init_status)); log::assert_that( init_status == std::future_status::ready, "Can't start stack, last instance: {}", registry_.last_instance_); log::info("init complete"); } void StackManager::handle_start_up(ModuleList* modules, Thread* stack_thread, std::promise promise) { registry_.Start(modules, stack_thread); promise.set_value(); } void StackManager::ShutDown() { WakelockManager::Get().Acquire(); std::promise promise; auto future = promise.get_future(); handler_->Post(common::BindOnce(&StackManager::handle_shut_down, common::Unretained(this), std::move(promise))); auto stop_status = future.wait_for(std::chrono::milliseconds( get_gd_stack_timeout_ms(/* is_start = */ false))); WakelockManager::Get().Release(); WakelockManager::Get().CleanUp(); log::assert_that( stop_status == std::future_status::ready, "Can't stop stack, last instance: {}", registry_.last_instance_); handler_->Clear(); handler_->WaitUntilStopped(std::chrono::milliseconds(2000)); delete handler_; delete management_thread_; } void StackManager::handle_shut_down(std::promise promise) { registry_.StopAll(); promise.set_value(); } std::chrono::milliseconds StackManager::get_gd_stack_timeout_ms(bool is_start) { auto gd_timeout = os::GetSystemPropertyUint32( is_start ? "bluetooth.gd.start_timeout" : "bluetooth.gd.stop_timeout", /* default_value = */ is_start ? 3000 : 5000); return std::chrono::milliseconds( gd_timeout * os::GetSystemPropertyUint32("ro.hw_timeout_multiplier", /* default_value = */ 1)); } } // namespace bluetooth