/* * 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 "module.h" using ::bluetooth::os::Handler; using ::bluetooth::os::Thread; namespace bluetooth { constexpr std::chrono::milliseconds kModuleStopTimeout = std::chrono::milliseconds(20); ModuleFactory::ModuleFactory(std::function ctor) : ctor_(ctor) { } std::string Module::ToString() const { return "Module"; } Handler* Module::GetHandler() const { ASSERT_LOG(handler_ != nullptr, "Can't get handler when it's not started"); return handler_; } const ModuleRegistry* Module::GetModuleRegistry() const { return registry_; } Module* Module::GetDependency(const ModuleFactory* module) const { for (auto& dependency : dependencies_.list_) { if (dependency == module) { return registry_->Get(module); } } ASSERT_LOG(false, "Module was not listed as a dependency in ListDependencies"); } Module* ModuleRegistry::Get(const ModuleFactory* module) const { auto instance = started_modules_.find(module); ASSERT(instance != started_modules_.end()); return instance->second; } bool ModuleRegistry::IsStarted(const ModuleFactory* module) const { return started_modules_.find(module) != started_modules_.end(); } void ModuleRegistry::Start(ModuleList* modules, Thread* thread) { for (auto it = modules->list_.begin(); it != modules->list_.end(); it++) { Start(*it, thread); } } void ModuleRegistry::set_registry_and_handler(Module* instance, Thread* thread) const { instance->registry_ = this; instance->handler_ = new Handler(thread); } Module* ModuleRegistry::Start(const ModuleFactory* module, Thread* thread) { auto started_instance = started_modules_.find(module); if (started_instance != started_modules_.end()) { return started_instance->second; } Module* instance = module->ctor_(); set_registry_and_handler(instance, thread); instance->ListDependencies(&instance->dependencies_); Start(&instance->dependencies_, thread); instance->Start(); start_order_.push_back(module); started_modules_[module] = instance; return instance; } void ModuleRegistry::StopAll() { // Since modules were brought up in dependency order, it is safe to tear down by going in reverse order. for (auto it = start_order_.rbegin(); it != start_order_.rend(); it++) { auto instance = started_modules_.find(*it); ASSERT(instance != started_modules_.end()); // Clear the handler before stopping the module to allow it to shut down gracefully. instance->second->handler_->Clear(); instance->second->handler_->WaitUntilStopped(kModuleStopTimeout); instance->second->Stop(); delete instance->second->handler_; delete instance->second; started_modules_.erase(instance); } ASSERT(started_modules_.empty()); start_order_.clear(); } os::Handler* ModuleRegistry::GetModuleHandler(const ModuleFactory* module) const { auto started_instance = started_modules_.find(module); if (started_instance != started_modules_.end()) { return started_instance->second->GetHandler(); } return nullptr; } } // namespace bluetooth