/* * Copyright (C) 2018 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 #include #include #include "ibinder_internal.h" #include "status_internal.h" using ::android::defaultServiceManager; using ::android::IBinder; using ::android::IServiceManager; using ::android::sp; using ::android::status_t; using ::android::statusToString; using ::android::String16; using ::android::String8; binder_exception_t AServiceManager_addService(AIBinder* binder, const char* instance) { if (binder == nullptr || instance == nullptr) { return EX_ILLEGAL_ARGUMENT; } sp sm = defaultServiceManager(); status_t exception = sm->addService(String16(instance), binder->getBinder()); return PruneException(exception); } binder_exception_t AServiceManager_addServiceWithFlags(AIBinder* binder, const char* instance, const AServiceManager_AddServiceFlag flags) { if (binder == nullptr || instance == nullptr) { return EX_ILLEGAL_ARGUMENT; } sp sm = defaultServiceManager(); bool allowIsolated = flags & AServiceManager_AddServiceFlag::ADD_SERVICE_ALLOW_ISOLATED; int dumpFlags = 0; if (flags & AServiceManager_AddServiceFlag::ADD_SERVICE_DUMP_FLAG_PRIORITY_CRITICAL) { dumpFlags |= IServiceManager::DUMP_FLAG_PRIORITY_CRITICAL; } if (flags & AServiceManager_AddServiceFlag::ADD_SERVICE_DUMP_FLAG_PRIORITY_HIGH) { dumpFlags |= IServiceManager::DUMP_FLAG_PRIORITY_HIGH; } if (flags & AServiceManager_AddServiceFlag::ADD_SERVICE_DUMP_FLAG_PRIORITY_NORMAL) { dumpFlags |= IServiceManager::DUMP_FLAG_PRIORITY_NORMAL; } if (flags & AServiceManager_AddServiceFlag::ADD_SERVICE_DUMP_FLAG_PRIORITY_DEFAULT) { dumpFlags |= IServiceManager::DUMP_FLAG_PRIORITY_DEFAULT; } if (dumpFlags == 0) { dumpFlags = IServiceManager::DUMP_FLAG_PRIORITY_DEFAULT; } status_t exception = sm->addService(String16(instance), binder->getBinder(), allowIsolated, dumpFlags); return PruneException(exception); } AIBinder* AServiceManager_checkService(const char* instance) { if (instance == nullptr) { return nullptr; } sp sm = defaultServiceManager(); sp binder = sm->checkService(String16(instance)); sp ret = ABpBinder::lookupOrCreateFromBinder(binder); AIBinder_incStrong(ret.get()); return ret.get(); } AIBinder* AServiceManager_getService(const char* instance) { if (instance == nullptr) { return nullptr; } sp sm = defaultServiceManager(); sp binder = sm->getService(String16(instance)); sp ret = ABpBinder::lookupOrCreateFromBinder(binder); AIBinder_incStrong(ret.get()); return ret.get(); } binder_status_t AServiceManager_registerLazyService(AIBinder* binder, const char* instance) { if (binder == nullptr || instance == nullptr) { return STATUS_UNEXPECTED_NULL; } auto serviceRegistrar = android::binder::LazyServiceRegistrar::getInstance(); status_t status = serviceRegistrar.registerService(binder->getBinder(), instance); return PruneStatusT(status); } AIBinder* AServiceManager_waitForService(const char* instance) { if (instance == nullptr) { return nullptr; } sp sm = defaultServiceManager(); sp binder = sm->waitForService(String16(instance)); sp ret = ABpBinder::lookupOrCreateFromBinder(binder); AIBinder_incStrong(ret.get()); return ret.get(); } typedef void (*AServiceManager_onRegister)(const char* instance, AIBinder* registered, void* cookie); struct AServiceManager_NotificationRegistration : public IServiceManager::LocalRegistrationCallback { std::mutex m; const char* instance = nullptr; void* cookie = nullptr; AServiceManager_onRegister onRegister = nullptr; virtual void onServiceRegistration(const String16& smInstance, const sp& binder) { std::lock_guard l(m); if (onRegister == nullptr) return; LOG_ALWAYS_FATAL_IF(String8(smInstance) != instance, "onServiceRegistration: %s != %s", String8(smInstance).c_str(), instance); sp ret = ABpBinder::lookupOrCreateFromBinder(binder); AIBinder_incStrong(ret.get()); onRegister(instance, ret.get(), cookie); } void clear() { std::lock_guard l(m); instance = nullptr; cookie = nullptr; onRegister = nullptr; } }; __attribute__((warn_unused_result)) AServiceManager_NotificationRegistration* AServiceManager_registerForServiceNotifications(const char* instance, AServiceManager_onRegister onRegister, void* cookie) { LOG_ALWAYS_FATAL_IF(instance == nullptr, "instance == nullptr"); LOG_ALWAYS_FATAL_IF(onRegister == nullptr, "onRegister == nullptr for %s", instance); // cookie can be nullptr auto cb = sp::make(); cb->instance = instance; cb->onRegister = onRegister; cb->cookie = cookie; sp sm = defaultServiceManager(); if (status_t res = sm->registerForNotifications(String16(instance), cb); res != STATUS_OK) { ALOGE("Failed to register for service notifications for %s: %s", instance, statusToString(res).c_str()); return nullptr; } cb->incStrong(nullptr); return cb.get(); } void AServiceManager_NotificationRegistration_delete( AServiceManager_NotificationRegistration* notification) { LOG_ALWAYS_FATAL_IF(notification == nullptr, "notification == nullptr"); notification->clear(); notification->decStrong(nullptr); } bool AServiceManager_isDeclared(const char* instance) { if (instance == nullptr) { return false; } sp sm = defaultServiceManager(); return sm->isDeclared(String16(instance)); } void AServiceManager_forEachDeclaredInstance(const char* interface, void* context, void (*callback)(const char*, void*)) { LOG_ALWAYS_FATAL_IF(interface == nullptr, "interface == nullptr"); // context may be nullptr LOG_ALWAYS_FATAL_IF(callback == nullptr, "callback == nullptr"); sp sm = defaultServiceManager(); for (const String16& instance : sm->getDeclaredInstances(String16(interface))) { callback(String8(instance).c_str(), context); } } bool AServiceManager_isUpdatableViaApex(const char* instance) { if (instance == nullptr) { return false; } sp sm = defaultServiceManager(); return sm->updatableViaApex(String16(instance)) != std::nullopt; } void AServiceManager_getUpdatableApexName(const char* instance, void* context, void (*callback)(const char*, void*)) { LOG_ALWAYS_FATAL_IF(instance == nullptr, "instance == nullptr"); // context may be nullptr LOG_ALWAYS_FATAL_IF(callback == nullptr, "callback == nullptr"); sp sm = defaultServiceManager(); std::optional updatableViaApex = sm->updatableViaApex(String16(instance)); if (updatableViaApex.has_value()) { callback(String8(updatableViaApex.value()).c_str(), context); } } void* AServiceManager_openDeclaredPassthroughHal(const char* interface, const char* instance, int flag) { LOG_ALWAYS_FATAL_IF(interface == nullptr, "interface == nullptr"); LOG_ALWAYS_FATAL_IF(instance == nullptr, "instance == nullptr"); return openDeclaredPassthroughHal(String16(interface), String16(instance), flag); } void AServiceManager_forceLazyServicesPersist(bool persist) { auto serviceRegistrar = android::binder::LazyServiceRegistrar::getInstance(); serviceRegistrar.forcePersist(persist); } void AServiceManager_setActiveServicesCallback(bool (*callback)(bool, void*), void* context) { auto serviceRegistrar = android::binder::LazyServiceRegistrar::getInstance(); std::function fn = [=](bool hasClients) -> bool { return callback(hasClients, context); }; serviceRegistrar.setActiveServicesCallback(fn); } bool AServiceManager_tryUnregister() { auto serviceRegistrar = android::binder::LazyServiceRegistrar::getInstance(); return serviceRegistrar.tryUnregister(); } void AServiceManager_reRegister() { auto serviceRegistrar = android::binder::LazyServiceRegistrar::getInstance(); serviceRegistrar.reRegister(); }