|  | /* | 
|  | * Copyright (C) 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. | 
|  | */ | 
|  |  | 
|  | #pragma once | 
|  |  | 
|  | #include <android/os/BnServiceManager.h> | 
|  | #include <android/os/IClientCallback.h> | 
|  | #include <android/os/IServiceCallback.h> | 
|  |  | 
|  | #include "Access.h" | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | using os::ConnectionInfo; | 
|  | using os::IClientCallback; | 
|  | using os::IServiceCallback; | 
|  | using os::ServiceDebugInfo; | 
|  |  | 
|  | class ServiceManager : public os::BnServiceManager, public IBinder::DeathRecipient { | 
|  | public: | 
|  | ServiceManager(std::unique_ptr<Access>&& access); | 
|  | ~ServiceManager(); | 
|  |  | 
|  | // getService will try to start any services it cannot find | 
|  | binder::Status getService(const std::string& name, sp<IBinder>* outBinder) override; | 
|  | binder::Status checkService(const std::string& name, sp<IBinder>* outBinder) override; | 
|  | binder::Status addService(const std::string& name, const sp<IBinder>& binder, | 
|  | bool allowIsolated, int32_t dumpPriority) override; | 
|  | binder::Status listServices(int32_t dumpPriority, std::vector<std::string>* outList) override; | 
|  | binder::Status registerForNotifications(const std::string& name, | 
|  | const sp<IServiceCallback>& callback) override; | 
|  | binder::Status unregisterForNotifications(const std::string& name, | 
|  | const sp<IServiceCallback>& callback) override; | 
|  |  | 
|  | binder::Status isDeclared(const std::string& name, bool* outReturn) override; | 
|  | binder::Status getDeclaredInstances(const std::string& interface, std::vector<std::string>* outReturn) override; | 
|  | binder::Status updatableViaApex(const std::string& name, | 
|  | std::optional<std::string>* outReturn) override; | 
|  | binder::Status getUpdatableNames(const std::string& apexName, | 
|  | std::vector<std::string>* outReturn) override; | 
|  | binder::Status getConnectionInfo(const std::string& name, | 
|  | std::optional<ConnectionInfo>* outReturn) override; | 
|  | binder::Status registerClientCallback(const std::string& name, const sp<IBinder>& service, | 
|  | const sp<IClientCallback>& cb) override; | 
|  | binder::Status tryUnregisterService(const std::string& name, const sp<IBinder>& binder) override; | 
|  | binder::Status getServiceDebugInfo(std::vector<ServiceDebugInfo>* outReturn) override; | 
|  | void binderDied(const wp<IBinder>& who) override; | 
|  | void handleClientCallbacks(); | 
|  |  | 
|  | /** | 
|  | *  This API is added for debug purposes. It clears members which hold service and callback | 
|  | * information. | 
|  | */ | 
|  | void clear(); | 
|  |  | 
|  | protected: | 
|  | virtual void tryStartService(const std::string& name); | 
|  |  | 
|  | private: | 
|  | struct Service { | 
|  | sp<IBinder> binder; // not null | 
|  | bool allowIsolated; | 
|  | int32_t dumpPriority; | 
|  | bool hasClients = false; // notifications sent on true -> false. | 
|  | bool guaranteeClient = false; // forces the client check to true | 
|  | Access::CallingContext ctx;   // process that originally registers this | 
|  |  | 
|  | // the number of clients of the service, including servicemanager itself | 
|  | ssize_t getNodeStrongRefCount(); | 
|  |  | 
|  | ~Service(); | 
|  | }; | 
|  |  | 
|  | using ServiceCallbackMap = std::map<std::string, std::vector<sp<IServiceCallback>>>; | 
|  | using ClientCallbackMap = std::map<std::string, std::vector<sp<IClientCallback>>>; | 
|  | using ServiceMap = std::map<std::string, Service>; | 
|  |  | 
|  | // removes a callback from mNameToRegistrationCallback, removing it if the vector is empty | 
|  | // this updates iterator to the next location | 
|  | void removeRegistrationCallback(const wp<IBinder>& who, | 
|  | ServiceCallbackMap::iterator* it, | 
|  | bool* found); | 
|  | // returns whether there are known clients in addition to the count provided | 
|  | bool handleServiceClientCallback(size_t knownClients, const std::string& serviceName, | 
|  | bool isCalledOnInterval); | 
|  | // Also updates mHasClients (of what the last callback was) | 
|  | void sendClientCallbackNotifications(const std::string& serviceName, bool hasClients, | 
|  | const char* context); | 
|  | // removes a callback from mNameToClientCallback, deleting the entry if the vector is empty | 
|  | // this updates the iterator to the next location | 
|  | void removeClientCallback(const wp<IBinder>& who, ClientCallbackMap::iterator* it); | 
|  |  | 
|  | sp<IBinder> tryGetService(const std::string& name, bool startIfNotFound); | 
|  |  | 
|  | ServiceMap mNameToService; | 
|  | ServiceCallbackMap mNameToRegistrationCallback; | 
|  | ClientCallbackMap mNameToClientCallback; | 
|  |  | 
|  | std::unique_ptr<Access> mAccess; | 
|  | }; | 
|  |  | 
|  | }  // namespace android |