1 /* 2 * Copyright 2019 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #pragma once 18 19 #include <functional> 20 #include <future> 21 #include <map> 22 #include <string> 23 #include <vector> 24 25 #include "common/bind.h" 26 #include "os/handler.h" 27 #include "os/log.h" 28 #include "os/thread.h" 29 30 namespace bluetooth { 31 32 class Module; 33 class ModuleRegistry; 34 35 class ModuleFactory { 36 friend ModuleRegistry; 37 public: 38 ModuleFactory(std::function<Module*()> ctor); 39 40 private: 41 std::function<Module*()> ctor_; 42 }; 43 44 class ModuleList { 45 friend ModuleRegistry; 46 friend Module; 47 public: 48 template <class T> 49 void add() { 50 list_.push_back(&T::Factory); 51 } 52 53 private: 54 std::vector<const ModuleFactory*> list_; 55 }; 56 57 // Each leaf node module must have a factory like so: 58 // 59 // static const ModuleFactory Factory; 60 // 61 // which will provide a constructor for the module registry to call. 62 // The module registry will also use the factory as the identifier 63 // for that module. 64 class Module { 65 friend ModuleRegistry; 66 public: 67 virtual ~Module() = default; 68 protected: 69 // Populate the provided list with modules that must start before yours 70 virtual void ListDependencies(ModuleList* list) = 0; 71 72 // You can grab your started dependencies during or after this call 73 // using GetDependency(), or access the module registry via GetModuleRegistry() 74 virtual void Start() = 0; 75 76 // Release all resources, you're about to be deleted 77 virtual void Stop() = 0; 78 79 virtual std::string ToString() const; 80 81 ::bluetooth::os::Handler* GetHandler() const; 82 83 const ModuleRegistry* GetModuleRegistry() const; 84 85 template <class T> 86 T* GetDependency() const { 87 return static_cast<T*>(GetDependency(&T::Factory)); 88 } 89 90 private: 91 Module* GetDependency(const ModuleFactory* module) const; 92 93 ::bluetooth::os::Handler* handler_ = nullptr; 94 ModuleList dependencies_; 95 const ModuleRegistry* registry_; 96 }; 97 98 class ModuleRegistry { 99 friend Module; 100 friend class StackManager; 101 public: 102 template <class T> 103 bool IsStarted() const { 104 return IsStarted(&T::Factory); 105 } 106 107 bool IsStarted(const ModuleFactory* factory) const; 108 109 // Start all the modules on this list and their dependencies 110 // in dependency order 111 void Start(ModuleList* modules, ::bluetooth::os::Thread* thread); 112 113 template <class T> 114 T* Start(::bluetooth::os::Thread* thread) { 115 return static_cast<T*>(Start(&T::Factory, thread)); 116 } 117 118 Module* Start(const ModuleFactory* id, ::bluetooth::os::Thread* thread); 119 120 // Stop all running modules in reverse order of start 121 void StopAll(); 122 123 protected: 124 Module* Get(const ModuleFactory* module) const; 125 126 void set_registry_and_handler(Module* instance, ::bluetooth::os::Thread* thread) const; 127 128 os::Handler* GetModuleHandler(const ModuleFactory* module) const; 129 130 std::map<const ModuleFactory*, Module*> started_modules_; 131 std::vector<const ModuleFactory*> start_order_; 132 }; 133 134 class TestModuleRegistry : public ModuleRegistry { 135 public: 136 void InjectTestModule(const ModuleFactory* module, Module* instance) { 137 start_order_.push_back(module); 138 started_modules_[module] = instance; 139 set_registry_and_handler(instance, &test_thread); 140 } 141 142 Module* GetModuleUnderTest(const ModuleFactory* module) const { 143 return Get(module); 144 } 145 146 os::Handler* GetTestModuleHandler(const ModuleFactory* module) const { 147 return GetModuleHandler(module); 148 } 149 150 os::Thread& GetTestThread() { 151 return test_thread; 152 } 153 154 bool SynchronizeModuleHandler(const ModuleFactory* module, std::chrono::milliseconds timeout) const { 155 std::promise<void> promise; 156 auto future = promise.get_future(); 157 os::Handler* handler = GetTestModuleHandler(module); 158 handler->Post(common::BindOnce(&std::promise<void>::set_value, common::Unretained(&promise))); 159 return future.wait_for(timeout) == std::future_status::ready; 160 } 161 162 private: 163 os::Thread test_thread{"test_thread", os::Thread::Priority::NORMAL}; 164 }; 165 166 } // namespace bluetooth 167