123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820 |
- #define LOG_TAG "hwservicemanager"
- #include "ServiceManager.h"
- #include "Vintf.h"
- #include <android-base/logging.h>
- #include <android-base/properties.h>
- #include <hwbinder/IPCThreadState.h>
- #include <hidl/HidlSupport.h>
- #include <hidl/HidlTransportSupport.h>
- #include <regex>
- #include <sstream>
- #include <thread>
- using android::hardware::IPCThreadState;
- using ::android::hardware::interfacesEqual;
- namespace android {
- namespace hidl {
- namespace manager {
- namespace implementation {
- AccessControl::CallingContext getBinderCallingContext() {
- const auto& self = IPCThreadState::self();
- pid_t pid = self->getCallingPid();
- const char* sid = self->getCallingSid();
- if (sid == nullptr) {
- if (pid != getpid()) {
- android_errorWriteLog(0x534e4554, "121035042");
- }
- return AccessControl::getCallingContext(pid);
- } else {
- return { true, sid, pid };
- }
- }
- static constexpr uint64_t kServiceDiedCookie = 0;
- static constexpr uint64_t kPackageListenerDiedCookie = 1;
- static constexpr uint64_t kServiceListenerDiedCookie = 2;
- static constexpr uint64_t kClientCallbackDiedCookie = 3;
- size_t ServiceManager::countExistingService() const {
- size_t total = 0;
- forEachExistingService([&] (const HidlService *) {
- ++total;
- return true;
- });
- return total;
- }
- void ServiceManager::forEachExistingService(std::function<bool(const HidlService *)> f) const {
- forEachServiceEntry([&] (const HidlService *service) {
- if (service->getService() == nullptr) {
- return true;
- }
- return f(service);
- });
- }
- void ServiceManager::forEachExistingService(std::function<bool(HidlService *)> f) {
- forEachServiceEntry([&] (HidlService *service) {
- if (service->getService() == nullptr) {
- return true;
- }
- return f(service);
- });
- }
- void ServiceManager::forEachServiceEntry(std::function<bool(const HidlService *)> f) const {
- for (const auto& interfaceMapping : mServiceMap) {
- const auto& instanceMap = interfaceMapping.second.getInstanceMap();
- for (const auto& instanceMapping : instanceMap) {
- if (!f(instanceMapping.second.get())) {
- return;
- }
- }
- }
- }
- void ServiceManager::forEachServiceEntry(std::function<bool(HidlService *)> f) {
- for (auto& interfaceMapping : mServiceMap) {
- auto& instanceMap = interfaceMapping.second.getInstanceMap();
- for (auto& instanceMapping : instanceMap) {
- if (!f(instanceMapping.second.get())) {
- return;
- }
- }
- }
- }
- HidlService* ServiceManager::lookup(const std::string& fqName, const std::string& name) {
- auto ifaceIt = mServiceMap.find(fqName);
- if (ifaceIt == mServiceMap.end()) {
- return nullptr;
- }
- PackageInterfaceMap &ifaceMap = ifaceIt->second;
- HidlService *hidlService = ifaceMap.lookup(name);
- return hidlService;
- }
- void ServiceManager::serviceDied(uint64_t cookie, const wp<IBase>& who) {
- bool serviceRemoved = false;
- switch (cookie) {
- case kServiceDiedCookie:
- serviceRemoved = removeService(who, nullptr );
- break;
- case kPackageListenerDiedCookie:
- serviceRemoved = removePackageListener(who);
- break;
- case kServiceListenerDiedCookie:
- serviceRemoved = removeServiceListener(who);
- break;
- case kClientCallbackDiedCookie: {
- sp<IBase> base = who.promote();
- IClientCallback* callback = static_cast<IClientCallback*>(base.get());
- serviceRemoved = unregisterClientCallback(nullptr ,
- sp<IClientCallback>(callback));
- } break;
- }
- if (!serviceRemoved) {
- LOG(ERROR) << "Received death notification but interface instance not removed. Cookie: "
- << cookie << " Service pointer: " << who.promote().get();
- }
- }
- ServiceManager::InstanceMap &ServiceManager::PackageInterfaceMap::getInstanceMap() {
- return mInstanceMap;
- }
- const ServiceManager::InstanceMap &ServiceManager::PackageInterfaceMap::getInstanceMap() const {
- return mInstanceMap;
- }
- const HidlService *ServiceManager::PackageInterfaceMap::lookup(
- const std::string &name) const {
- auto it = mInstanceMap.find(name);
- if (it == mInstanceMap.end()) {
- return nullptr;
- }
- return it->second.get();
- }
- HidlService *ServiceManager::PackageInterfaceMap::lookup(
- const std::string &name) {
- return const_cast<HidlService*>(
- const_cast<const PackageInterfaceMap*>(this)->lookup(name));
- }
- void ServiceManager::PackageInterfaceMap::insertService(
- std::unique_ptr<HidlService> &&service) {
- mInstanceMap.insert({service->getInstanceName(), std::move(service)});
- }
- void ServiceManager::PackageInterfaceMap::sendPackageRegistrationNotification(
- const hidl_string &fqName,
- const hidl_string &instanceName) {
- for (auto it = mPackageListeners.begin(); it != mPackageListeners.end();) {
- auto ret = (*it)->onRegistration(fqName, instanceName, false );
- if (ret.isOk()) {
- ++it;
- } else {
- LOG(ERROR) << "Dropping registration callback for " << fqName << "/" << instanceName
- << ": transport error.";
- it = mPackageListeners.erase(it);
- }
- }
- }
- void ServiceManager::PackageInterfaceMap::addPackageListener(sp<IServiceNotification> listener) {
- for (const auto &instanceMapping : mInstanceMap) {
- const std::unique_ptr<HidlService> &service = instanceMapping.second;
- if (service->getService() == nullptr) {
- continue;
- }
- auto ret = listener->onRegistration(
- service->getInterfaceName(),
- service->getInstanceName(),
- true );
- if (!ret.isOk()) {
- LOG(ERROR) << "Not adding package listener for " << service->getInterfaceName()
- << "/" << service->getInstanceName() << ": transport error "
- << "when sending notification for already registered instance.";
- return;
- }
- }
- mPackageListeners.push_back(listener);
- }
- bool ServiceManager::PackageInterfaceMap::removePackageListener(const wp<IBase>& who) {
- bool found = false;
- for (auto it = mPackageListeners.begin(); it != mPackageListeners.end();) {
- if (interfacesEqual(*it, who.promote())) {
- it = mPackageListeners.erase(it);
- found = true;
- } else {
- ++it;
- }
- }
- return found;
- }
- bool ServiceManager::PackageInterfaceMap::removeServiceListener(const wp<IBase>& who) {
- bool found = false;
- for (auto &servicePair : getInstanceMap()) {
- const std::unique_ptr<HidlService> &service = servicePair.second;
- found |= service->removeListener(who);
- }
- return found;
- }
- static void tryStartService(const std::string& fqName, const std::string& name) {
- using ::android::base::SetProperty;
- std::thread([=] {
- bool success = SetProperty("ctl.interface_start", fqName + "/" + name);
- if (!success) {
- LOG(ERROR) << "Failed to set property for starting " << fqName << "/" << name;
- }
- }).detach();
- }
- Return<sp<IBase>> ServiceManager::get(const hidl_string& hidlFqName,
- const hidl_string& hidlName) {
- const std::string fqName = hidlFqName;
- const std::string name = hidlName;
- if (!mAcl.canGet(fqName, getBinderCallingContext())) {
- return nullptr;
- }
- HidlService* hidlService = lookup(fqName, name);
- if (hidlService == nullptr) {
- tryStartService(fqName, name);
- return nullptr;
- }
- sp<IBase> service = hidlService->getService();
- if (service == nullptr) {
- tryStartService(fqName, name);
- return nullptr;
- }
-
-
- hidlService->guaranteeClient();
-
-
-
-
-
- hardware::addPostCommandTask([hidlService] {
- hidlService->handleClientCallbacks(false );
- });
- return service;
- }
- Return<bool> ServiceManager::add(const hidl_string& name, const sp<IBase>& service) {
- bool addSuccess = false;
- if (service == nullptr) {
- return false;
- }
- auto pidcon = getBinderCallingContext();
- auto ret = service->interfaceChain([&](const auto &interfaceChain) {
- addSuccess = addImpl(name, service, interfaceChain, pidcon);
- });
- if (!ret.isOk()) {
- LOG(ERROR) << "Failed to retrieve interface chain: " << ret.description();
- return false;
- }
- return addSuccess;
- }
- bool ServiceManager::addImpl(const std::string& name,
- const sp<IBase>& service,
- const hidl_vec<hidl_string>& interfaceChain,
- const AccessControl::CallingContext& callingContext) {
- if (interfaceChain.size() == 0) {
- LOG(WARNING) << "Empty interface chain for " << name;
- return false;
- }
-
- for(size_t i = 0; i < interfaceChain.size(); i++) {
- const std::string fqName = interfaceChain[i];
- if (!mAcl.canAdd(fqName, callingContext)) {
- return false;
- }
- }
-
- if (interfaceChain.size() > 1) {
-
- const std::string baseFqName = interfaceChain[interfaceChain.size() - 2];
- const HidlService *hidlService = lookup(baseFqName, name);
- if (hidlService != nullptr && hidlService->getService() != nullptr) {
-
-
-
-
-
- const std::string childFqName = interfaceChain[0];
- pid_t newServicePid = IPCThreadState::self()->getCallingPid();
- pid_t oldServicePid = hidlService->getDebugPid();
- LOG(WARNING) << "Detected instance of " << childFqName << " (pid: " << newServicePid
- << ") registering over instance of or with base of " << baseFqName << " (pid: "
- << oldServicePid << ").";
- }
- }
-
- {
-
-
-
-
-
-
- const std::string childFqName = interfaceChain[0];
- const HidlService *hidlService = lookup(childFqName, name);
- if (hidlService != nullptr) {
- const sp<IBase> remove = hidlService->getService();
- if (remove != nullptr) {
- const std::string instanceName = name;
- removeService(remove, &instanceName );
- }
- }
- }
- for(size_t i = 0; i < interfaceChain.size(); i++) {
- const std::string fqName = interfaceChain[i];
- PackageInterfaceMap &ifaceMap = mServiceMap[fqName];
- HidlService *hidlService = ifaceMap.lookup(name);
- if (hidlService == nullptr) {
- ifaceMap.insertService(
- std::make_unique<HidlService>(fqName, name, service, callingContext.pid));
- } else {
- hidlService->setService(service, callingContext.pid);
- }
- ifaceMap.sendPackageRegistrationNotification(fqName, name);
- }
- bool linkRet = service->linkToDeath(this, kServiceDiedCookie).withDefault(false);
- if (!linkRet) {
- LOG(ERROR) << "Could not link to death for " << interfaceChain[0] << "/" << name;
- }
- return true;
- }
- Return<ServiceManager::Transport> ServiceManager::getTransport(const hidl_string& fqName,
- const hidl_string& name) {
- using ::android::hardware::getTransport;
- if (!mAcl.canGet(fqName, getBinderCallingContext())) {
- return Transport::EMPTY;
- }
- switch (getTransport(fqName, name)) {
- case vintf::Transport::HWBINDER:
- return Transport::HWBINDER;
- case vintf::Transport::PASSTHROUGH:
- return Transport::PASSTHROUGH;
- case vintf::Transport::EMPTY:
- default:
- return Transport::EMPTY;
- }
- }
- Return<void> ServiceManager::list(list_cb _hidl_cb) {
- if (!mAcl.canList(getBinderCallingContext())) {
- _hidl_cb({});
- return Void();
- }
- hidl_vec<hidl_string> list;
- list.resize(countExistingService());
- size_t idx = 0;
- forEachExistingService([&] (const HidlService *service) {
- list[idx++] = service->string();
- return true;
- });
- _hidl_cb(list);
- return Void();
- }
- Return<void> ServiceManager::listByInterface(const hidl_string& fqName,
- listByInterface_cb _hidl_cb) {
- if (!mAcl.canGet(fqName, getBinderCallingContext())) {
- _hidl_cb({});
- return Void();
- }
- auto ifaceIt = mServiceMap.find(fqName);
- if (ifaceIt == mServiceMap.end()) {
- _hidl_cb(hidl_vec<hidl_string>());
- return Void();
- }
- const auto &instanceMap = ifaceIt->second.getInstanceMap();
- hidl_vec<hidl_string> list;
- size_t total = 0;
- for (const auto &serviceMapping : instanceMap) {
- const std::unique_ptr<HidlService> &service = serviceMapping.second;
- if (service->getService() == nullptr) continue;
- ++total;
- }
- list.resize(total);
- size_t idx = 0;
- for (const auto &serviceMapping : instanceMap) {
- const std::unique_ptr<HidlService> &service = serviceMapping.second;
- if (service->getService() == nullptr) continue;
- list[idx++] = service->getInstanceName();
- }
- _hidl_cb(list);
- return Void();
- }
- Return<bool> ServiceManager::registerForNotifications(const hidl_string& fqName,
- const hidl_string& name,
- const sp<IServiceNotification>& callback) {
- if (callback == nullptr) {
- return false;
- }
- if (!mAcl.canGet(fqName, getBinderCallingContext())) {
- return false;
- }
- PackageInterfaceMap &ifaceMap = mServiceMap[fqName];
- if (name.empty()) {
- bool ret = callback->linkToDeath(this, kPackageListenerDiedCookie).withDefault(false);
- if (!ret) {
- LOG(ERROR) << "Failed to register death recipient for " << fqName << "/" << name;
- return false;
- }
- ifaceMap.addPackageListener(callback);
- return true;
- }
- HidlService *service = ifaceMap.lookup(name);
- bool ret = callback->linkToDeath(this, kServiceListenerDiedCookie).withDefault(false);
- if (!ret) {
- LOG(ERROR) << "Failed to register death recipient for " << fqName << "/" << name;
- return false;
- }
- if (service == nullptr) {
- auto adding = std::make_unique<HidlService>(fqName, name);
- adding->addListener(callback);
- ifaceMap.insertService(std::move(adding));
- } else {
- service->addListener(callback);
- }
- return true;
- }
- Return<bool> ServiceManager::unregisterForNotifications(const hidl_string& fqName,
- const hidl_string& name,
- const sp<IServiceNotification>& callback) {
- if (callback == nullptr) {
- LOG(ERROR) << "Cannot unregister null callback for " << fqName << "/" << name;
- return false;
- }
-
-
- if (fqName.empty()) {
- bool success = false;
- success |= removePackageListener(callback);
- success |= removeServiceListener(callback);
- return success;
- }
- PackageInterfaceMap &ifaceMap = mServiceMap[fqName];
- if (name.empty()) {
- bool success = false;
- success |= ifaceMap.removePackageListener(callback);
- success |= ifaceMap.removeServiceListener(callback);
- return success;
- }
- HidlService *service = ifaceMap.lookup(name);
- if (service == nullptr) {
- return false;
- }
- return service->removeListener(callback);
- }
- Return<bool> ServiceManager::registerClientCallback(const hidl_string& hidlFqName,
- const hidl_string& hidlName,
- const sp<IBase>& server,
- const sp<IClientCallback>& cb) {
- if (server == nullptr || cb == nullptr) return false;
- const std::string fqName = hidlFqName;
- const std::string name = hidlName;
-
- pid_t pid = IPCThreadState::self()->getCallingPid();
- if (!mAcl.canAdd(fqName, getBinderCallingContext())) {
- return false;
- }
- HidlService* registered = lookup(fqName, name);
- if (registered == nullptr) {
- return false;
- }
-
- if (registered->getDebugPid() != pid) {
- LOG(WARNING) << "Only a server can register for client callbacks (for " << fqName
- << "/" << name << ")";
- return false;
- }
- sp<IBase> service = registered->getService();
- if (!interfacesEqual(service, server)) {
- LOG(WARNING) << "Tried to register client callback for " << fqName << "/" << name
- << " but a different service is registered under this name.";
- return false;
- }
- bool linkRet = cb->linkToDeath(this, kClientCallbackDiedCookie).withDefault(false);
- if (!linkRet) {
- LOG(ERROR) << "Could not link to death for registerClientCallback";
- return false;
- }
- registered->addClientCallback(cb);
- return true;
- }
- Return<bool> ServiceManager::unregisterClientCallback(const sp<IBase>& server,
- const sp<IClientCallback>& cb) {
- if (cb == nullptr) return false;
- bool removed = false;
- forEachExistingService([&] (HidlService *service) {
- if (server == nullptr || interfacesEqual(service->getService(), server)) {
- removed |= service->removeClientCallback(cb);
- }
- return true;
- });
- return removed;
- }
- void ServiceManager::handleClientCallbacks() {
- forEachServiceEntry([&] (HidlService *service) {
- service->handleClientCallbacks(true );
- return true;
- });
- }
- Return<bool> ServiceManager::addWithChain(const hidl_string& name,
- const sp<IBase>& service,
- const hidl_vec<hidl_string>& chain) {
- if (service == nullptr) {
- return false;
- }
- auto callingContext = getBinderCallingContext();
- return addImpl(name, service, chain, callingContext);
- }
- Return<void> ServiceManager::listManifestByInterface(const hidl_string& fqName,
- listManifestByInterface_cb _hidl_cb) {
- if (!mAcl.canGet(fqName, getBinderCallingContext())) {
- _hidl_cb({});
- return Void();
- }
- std::set<std::string> instances = getInstances(fqName);
- hidl_vec<hidl_string> ret(instances.begin(), instances.end());
- _hidl_cb(ret);
- return Void();
- }
- Return<bool> ServiceManager::tryUnregister(const hidl_string& hidlFqName,
- const hidl_string& hidlName,
- const sp<IBase>& service) {
- const std::string fqName = hidlFqName;
- const std::string name = hidlName;
- if (service == nullptr) {
- return false;
- }
- if (!mAcl.canAdd(fqName, getBinderCallingContext())) {
- return false;
- }
- HidlService* registered = lookup(fqName, name);
-
- pid_t pid = IPCThreadState::self()->getCallingPid();
- if (registered->getDebugPid() != pid) {
- LOG(WARNING) << "Only a server can unregister itself (for " << fqName
- << "/" << name << ")";
- return false;
- }
- sp<IBase> server = registered->getService();
- if (!interfacesEqual(service, server)) {
- LOG(WARNING) << "Tried to unregister for " << fqName << "/" << name
- << " but a different service is registered under this name.";
- return false;
- }
- int clients = registered->forceHandleClientCallbacks(false );
-
-
-
-
-
- if (clients < 0 || clients > 2) {
-
- LOG(INFO) << "Tried to unregister for " << fqName << "/" << name
- << " but there are clients: " << clients;
- return false;
- }
-
- bool success = removeService(service, &name );
- if (registered->getService() != nullptr) {
- LOG(ERROR) << "Bad state. Unregistration failed for " << fqName << "/" << name << ".";
- return false;
- }
- return success;
- }
- Return<void> ServiceManager::debugDump(debugDump_cb _cb) {
- if (!mAcl.canList(getBinderCallingContext())) {
- _cb({});
- return Void();
- }
- std::vector<IServiceManager::InstanceDebugInfo> list;
- forEachServiceEntry([&] (const HidlService *service) {
- hidl_vec<int32_t> clientPids;
- clientPids.resize(service->getPassthroughClients().size());
- size_t i = 0;
- for (pid_t p : service->getPassthroughClients()) {
- clientPids[i++] = p;
- }
- list.push_back({
- .pid = service->getDebugPid(),
- .interfaceName = service->getInterfaceName(),
- .instanceName = service->getInstanceName(),
- .clientPids = clientPids,
- .arch = ::android::hidl::base::V1_0::DebugInfo::Architecture::UNKNOWN
- });
- return true;
- });
- _cb(list);
- return Void();
- }
- Return<void> ServiceManager::registerPassthroughClient(const hidl_string &fqName,
- const hidl_string &name) {
- auto callingContext = getBinderCallingContext();
- if (!mAcl.canGet(fqName, callingContext)) {
-
- return Void();
- }
- PackageInterfaceMap &ifaceMap = mServiceMap[fqName];
- if (name.empty()) {
- LOG(WARNING) << "registerPassthroughClient encounters empty instance name for "
- << fqName.c_str();
- return Void();
- }
- HidlService *service = ifaceMap.lookup(name);
- if (service == nullptr) {
- auto adding = std::make_unique<HidlService>(fqName, name);
- adding->registerPassthroughClient(callingContext.pid);
- ifaceMap.insertService(std::move(adding));
- } else {
- service->registerPassthroughClient(callingContext.pid);
- }
- return Void();
- }
- bool ServiceManager::removeService(const wp<IBase>& who, const std::string* restrictToInstanceName) {
- bool keepInstance = false;
- bool removed = false;
- for (auto &interfaceMapping : mServiceMap) {
- auto &instanceMap = interfaceMapping.second.getInstanceMap();
- for (auto &servicePair : instanceMap) {
- const std::string &instanceName = servicePair.first;
- const std::unique_ptr<HidlService> &service = servicePair.second;
- if (interfacesEqual(service->getService(), who.promote())) {
- if (restrictToInstanceName != nullptr && *restrictToInstanceName != instanceName) {
-
-
- keepInstance = true;
- continue;
- }
- service->setService(nullptr, static_cast<pid_t>(IServiceManager::PidConstant::NO_PID));
- removed = true;
- }
- }
- }
- return !keepInstance && removed;
- }
- bool ServiceManager::removePackageListener(const wp<IBase>& who) {
- bool found = false;
- for (auto &interfaceMapping : mServiceMap) {
- found |= interfaceMapping.second.removePackageListener(who);
- }
- return found;
- }
- bool ServiceManager::removeServiceListener(const wp<IBase>& who) {
- bool found = false;
- for (auto &interfaceMapping : mServiceMap) {
- auto &packageInterfaceMap = interfaceMapping.second;
- found |= packageInterfaceMap.removeServiceListener(who);
- }
- return found;
- }
- }
- }
- }
- }
|