xref: /aosp_15_r20/system/hwservicemanager/service.cpp (revision ee3b7b6295061e544d3520b965ea91a90424af41)
1*ee3b7b62SAndroid Build Coastguard Worker /*
2*ee3b7b62SAndroid Build Coastguard Worker  * Copyright (C) 2016 The Android Open Source Project
3*ee3b7b62SAndroid Build Coastguard Worker  *
4*ee3b7b62SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*ee3b7b62SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*ee3b7b62SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*ee3b7b62SAndroid Build Coastguard Worker  *
8*ee3b7b62SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*ee3b7b62SAndroid Build Coastguard Worker  *
10*ee3b7b62SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*ee3b7b62SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*ee3b7b62SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ee3b7b62SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*ee3b7b62SAndroid Build Coastguard Worker  * limitations under the License.
15*ee3b7b62SAndroid Build Coastguard Worker  */
16*ee3b7b62SAndroid Build Coastguard Worker 
17*ee3b7b62SAndroid Build Coastguard Worker #define LOG_TAG "hwservicemanager"
18*ee3b7b62SAndroid Build Coastguard Worker 
19*ee3b7b62SAndroid Build Coastguard Worker #include <utils/Log.h>
20*ee3b7b62SAndroid Build Coastguard Worker 
21*ee3b7b62SAndroid Build Coastguard Worker #include <inttypes.h>
22*ee3b7b62SAndroid Build Coastguard Worker #include <unistd.h>
23*ee3b7b62SAndroid Build Coastguard Worker #include <sys/timerfd.h>
24*ee3b7b62SAndroid Build Coastguard Worker 
25*ee3b7b62SAndroid Build Coastguard Worker #include <android/hidl/token/1.0/ITokenManager.h>
26*ee3b7b62SAndroid Build Coastguard Worker #include <cutils/properties.h>
27*ee3b7b62SAndroid Build Coastguard Worker #include <hidl/HidlBinderSupport.h>
28*ee3b7b62SAndroid Build Coastguard Worker #include <hidl/HidlTransportSupport.h>
29*ee3b7b62SAndroid Build Coastguard Worker #include <hidl/ServiceManagement.h>
30*ee3b7b62SAndroid Build Coastguard Worker #include <hidl/Status.h>
31*ee3b7b62SAndroid Build Coastguard Worker #include <hwbinder/IPCThreadState.h>
32*ee3b7b62SAndroid Build Coastguard Worker #include <hwbinder/ProcessState.h>
33*ee3b7b62SAndroid Build Coastguard Worker #include <utils/Errors.h>
34*ee3b7b62SAndroid Build Coastguard Worker #include <utils/Looper.h>
35*ee3b7b62SAndroid Build Coastguard Worker #include <utils/StrongPointer.h>
36*ee3b7b62SAndroid Build Coastguard Worker 
37*ee3b7b62SAndroid Build Coastguard Worker #include "ServiceManager.h"
38*ee3b7b62SAndroid Build Coastguard Worker #include "TokenManager.h"
39*ee3b7b62SAndroid Build Coastguard Worker #include "Vintf.h"
40*ee3b7b62SAndroid Build Coastguard Worker 
41*ee3b7b62SAndroid Build Coastguard Worker // libutils:
42*ee3b7b62SAndroid Build Coastguard Worker using android::sp;
43*ee3b7b62SAndroid Build Coastguard Worker using android::Looper;
44*ee3b7b62SAndroid Build Coastguard Worker using android::LooperCallback;
45*ee3b7b62SAndroid Build Coastguard Worker 
46*ee3b7b62SAndroid Build Coastguard Worker // libhwbinder:
47*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::BHwBinder;
48*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::IBinder;
49*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::IPCThreadState;
50*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::ProcessState;
51*ee3b7b62SAndroid Build Coastguard Worker 
52*ee3b7b62SAndroid Build Coastguard Worker // libhidl
53*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::handleTransportPoll;
54*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::setRequestingSid;
55*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::HidlReturnRestriction;
56*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::setProcessHidlReturnRestriction;
57*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::setupTransportPolling;
58*ee3b7b62SAndroid Build Coastguard Worker using android::hardware::toBinder;
59*ee3b7b62SAndroid Build Coastguard Worker 
60*ee3b7b62SAndroid Build Coastguard Worker // implementations
61*ee3b7b62SAndroid Build Coastguard Worker using android::hidl::manager::implementation::ServiceManager;
62*ee3b7b62SAndroid Build Coastguard Worker using android::hidl::manager::V1_0::IServiceManager;
63*ee3b7b62SAndroid Build Coastguard Worker using android::hidl::token::V1_0::implementation::TokenManager;
64*ee3b7b62SAndroid Build Coastguard Worker 
65*ee3b7b62SAndroid Build Coastguard Worker static std::string serviceName = "default";
66*ee3b7b62SAndroid Build Coastguard Worker 
67*ee3b7b62SAndroid Build Coastguard Worker class HwBinderCallback : public LooperCallback {
68*ee3b7b62SAndroid Build Coastguard Worker public:
setupTo(const sp<Looper> & looper)69*ee3b7b62SAndroid Build Coastguard Worker     static sp<HwBinderCallback> setupTo(const sp<Looper>& looper) {
70*ee3b7b62SAndroid Build Coastguard Worker         sp<HwBinderCallback> cb = new HwBinderCallback;
71*ee3b7b62SAndroid Build Coastguard Worker 
72*ee3b7b62SAndroid Build Coastguard Worker         int fdHwBinder = setupTransportPolling();
73*ee3b7b62SAndroid Build Coastguard Worker         LOG_ALWAYS_FATAL_IF(fdHwBinder < 0, "Failed to setupTransportPolling: %d", fdHwBinder);
74*ee3b7b62SAndroid Build Coastguard Worker 
75*ee3b7b62SAndroid Build Coastguard Worker         // Flush after setupPolling(), to make sure the binder driver
76*ee3b7b62SAndroid Build Coastguard Worker         // knows about this thread handling commands.
77*ee3b7b62SAndroid Build Coastguard Worker         IPCThreadState::self()->flushCommands();
78*ee3b7b62SAndroid Build Coastguard Worker 
79*ee3b7b62SAndroid Build Coastguard Worker         int ret = looper->addFd(fdHwBinder,
80*ee3b7b62SAndroid Build Coastguard Worker                                 Looper::POLL_CALLBACK,
81*ee3b7b62SAndroid Build Coastguard Worker                                 Looper::EVENT_INPUT,
82*ee3b7b62SAndroid Build Coastguard Worker                                 cb,
83*ee3b7b62SAndroid Build Coastguard Worker                                 nullptr /*data*/);
84*ee3b7b62SAndroid Build Coastguard Worker         LOG_ALWAYS_FATAL_IF(ret != 1, "Failed to add binder FD to Looper");
85*ee3b7b62SAndroid Build Coastguard Worker 
86*ee3b7b62SAndroid Build Coastguard Worker         return cb;
87*ee3b7b62SAndroid Build Coastguard Worker     }
88*ee3b7b62SAndroid Build Coastguard Worker 
handleEvent(int fd,int,void *)89*ee3b7b62SAndroid Build Coastguard Worker     int handleEvent(int fd, int /*events*/, void* /*data*/) override {
90*ee3b7b62SAndroid Build Coastguard Worker         handleTransportPoll(fd);
91*ee3b7b62SAndroid Build Coastguard Worker         return 1;  // Continue receiving callbacks.
92*ee3b7b62SAndroid Build Coastguard Worker     }
93*ee3b7b62SAndroid Build Coastguard Worker };
94*ee3b7b62SAndroid Build Coastguard Worker 
95*ee3b7b62SAndroid Build Coastguard Worker // LooperCallback for IClientCallback
96*ee3b7b62SAndroid Build Coastguard Worker class ClientCallbackCallback : public LooperCallback {
97*ee3b7b62SAndroid Build Coastguard Worker public:
setupTo(const sp<Looper> & looper,const sp<ServiceManager> & manager)98*ee3b7b62SAndroid Build Coastguard Worker     static sp<ClientCallbackCallback> setupTo(const sp<Looper>& looper, const sp<ServiceManager>& manager) {
99*ee3b7b62SAndroid Build Coastguard Worker         sp<ClientCallbackCallback> cb = new ClientCallbackCallback(manager);
100*ee3b7b62SAndroid Build Coastguard Worker 
101*ee3b7b62SAndroid Build Coastguard Worker         int fdTimer = timerfd_create(CLOCK_MONOTONIC, 0 /*flags*/);
102*ee3b7b62SAndroid Build Coastguard Worker         LOG_ALWAYS_FATAL_IF(fdTimer < 0, "Failed to timerfd_create: fd: %d err: %d", fdTimer, errno);
103*ee3b7b62SAndroid Build Coastguard Worker 
104*ee3b7b62SAndroid Build Coastguard Worker         itimerspec timespec {
105*ee3b7b62SAndroid Build Coastguard Worker             .it_interval = {
106*ee3b7b62SAndroid Build Coastguard Worker                 .tv_sec = 5,
107*ee3b7b62SAndroid Build Coastguard Worker                 .tv_nsec = 0,
108*ee3b7b62SAndroid Build Coastguard Worker             },
109*ee3b7b62SAndroid Build Coastguard Worker             .it_value = {
110*ee3b7b62SAndroid Build Coastguard Worker                 .tv_sec = 5,
111*ee3b7b62SAndroid Build Coastguard Worker                 .tv_nsec = 0,
112*ee3b7b62SAndroid Build Coastguard Worker             },
113*ee3b7b62SAndroid Build Coastguard Worker         };
114*ee3b7b62SAndroid Build Coastguard Worker 
115*ee3b7b62SAndroid Build Coastguard Worker         int timeRes = timerfd_settime(fdTimer, 0 /*flags*/, &timespec, nullptr);
116*ee3b7b62SAndroid Build Coastguard Worker         LOG_ALWAYS_FATAL_IF(timeRes < 0, "Failed to timerfd_settime: res: %d err: %d", timeRes, errno);
117*ee3b7b62SAndroid Build Coastguard Worker 
118*ee3b7b62SAndroid Build Coastguard Worker         int addRes = looper->addFd(fdTimer,
119*ee3b7b62SAndroid Build Coastguard Worker                                    Looper::POLL_CALLBACK,
120*ee3b7b62SAndroid Build Coastguard Worker                                    Looper::EVENT_INPUT,
121*ee3b7b62SAndroid Build Coastguard Worker                                    cb,
122*ee3b7b62SAndroid Build Coastguard Worker                                    nullptr);
123*ee3b7b62SAndroid Build Coastguard Worker         LOG_ALWAYS_FATAL_IF(addRes != 1, "Failed to add client callback FD to Looper");
124*ee3b7b62SAndroid Build Coastguard Worker 
125*ee3b7b62SAndroid Build Coastguard Worker         return cb;
126*ee3b7b62SAndroid Build Coastguard Worker     }
127*ee3b7b62SAndroid Build Coastguard Worker 
handleEvent(int fd,int,void *)128*ee3b7b62SAndroid Build Coastguard Worker     int handleEvent(int fd, int /*events*/, void* /*data*/) override {
129*ee3b7b62SAndroid Build Coastguard Worker         uint64_t expirations;
130*ee3b7b62SAndroid Build Coastguard Worker         int ret = read(fd, &expirations, sizeof(expirations));
131*ee3b7b62SAndroid Build Coastguard Worker         if (ret != sizeof(expirations)) {
132*ee3b7b62SAndroid Build Coastguard Worker             ALOGE("Read failed to callback FD: ret: %d err: %d", ret, errno);
133*ee3b7b62SAndroid Build Coastguard Worker         }
134*ee3b7b62SAndroid Build Coastguard Worker 
135*ee3b7b62SAndroid Build Coastguard Worker         mManager->handleClientCallbacks();
136*ee3b7b62SAndroid Build Coastguard Worker         return 1;  // Continue receiving callbacks.
137*ee3b7b62SAndroid Build Coastguard Worker     }
138*ee3b7b62SAndroid Build Coastguard Worker private:
ClientCallbackCallback(const sp<ServiceManager> & manager)139*ee3b7b62SAndroid Build Coastguard Worker     ClientCallbackCallback(const sp<ServiceManager>& manager) : mManager(manager) {}
140*ee3b7b62SAndroid Build Coastguard Worker     sp<ServiceManager> mManager;
141*ee3b7b62SAndroid Build Coastguard Worker };
142*ee3b7b62SAndroid Build Coastguard Worker 
main()143*ee3b7b62SAndroid Build Coastguard Worker int main() {
144*ee3b7b62SAndroid Build Coastguard Worker     // If hwservicemanager crashes, the system may be unstable and hard to debug. This is both why
145*ee3b7b62SAndroid Build Coastguard Worker     // we log this and why we care about this at all.
146*ee3b7b62SAndroid Build Coastguard Worker     setProcessHidlReturnRestriction(HidlReturnRestriction::ERROR_IF_UNCHECKED);
147*ee3b7b62SAndroid Build Coastguard Worker 
148*ee3b7b62SAndroid Build Coastguard Worker     // TODO(b/36424585): make fatal
149*ee3b7b62SAndroid Build Coastguard Worker     ProcessState::self()->setCallRestriction(ProcessState::CallRestriction::ERROR_IF_NOT_ONEWAY);
150*ee3b7b62SAndroid Build Coastguard Worker     auto transport = android::hardware::getTransport(ServiceManager::descriptor, serviceName);
151*ee3b7b62SAndroid Build Coastguard Worker     if (transport == android::vintf::Transport::EMPTY) {
152*ee3b7b62SAndroid Build Coastguard Worker         ALOGI("HIDL is not supported on this device so hwservicemanager is not needed");
153*ee3b7b62SAndroid Build Coastguard Worker         int rc = property_set("hwservicemanager.disabled", "true");
154*ee3b7b62SAndroid Build Coastguard Worker         if (rc) {
155*ee3b7b62SAndroid Build Coastguard Worker             LOG_ALWAYS_FATAL("Failed to set \"hwservicemanager.disabled\" (error %d).\"", rc);
156*ee3b7b62SAndroid Build Coastguard Worker         }
157*ee3b7b62SAndroid Build Coastguard Worker         // wait here for init to see the proprty and shut us down
158*ee3b7b62SAndroid Build Coastguard Worker         while (true) {
159*ee3b7b62SAndroid Build Coastguard Worker             ALOGW("Waiting on init to shut this process down.");
160*ee3b7b62SAndroid Build Coastguard Worker             sleep(10);
161*ee3b7b62SAndroid Build Coastguard Worker         }
162*ee3b7b62SAndroid Build Coastguard Worker     }
163*ee3b7b62SAndroid Build Coastguard Worker 
164*ee3b7b62SAndroid Build Coastguard Worker     sp<ServiceManager> manager = new ServiceManager();
165*ee3b7b62SAndroid Build Coastguard Worker     setRequestingSid(manager, true);
166*ee3b7b62SAndroid Build Coastguard Worker 
167*ee3b7b62SAndroid Build Coastguard Worker     if (!manager->add(serviceName, manager).withDefault(false)) {
168*ee3b7b62SAndroid Build Coastguard Worker         ALOGE("Failed to register hwservicemanager with itself.");
169*ee3b7b62SAndroid Build Coastguard Worker     }
170*ee3b7b62SAndroid Build Coastguard Worker 
171*ee3b7b62SAndroid Build Coastguard Worker     // Check to make sure we should be registering tokenManager first. Only if
172*ee3b7b62SAndroid Build Coastguard Worker     // it's declared in the manifest.
173*ee3b7b62SAndroid Build Coastguard Worker     sp<TokenManager> tokenManager;
174*ee3b7b62SAndroid Build Coastguard Worker     if (android::vintf::Transport::EMPTY !=
175*ee3b7b62SAndroid Build Coastguard Worker         android::hardware::getTransport(TokenManager::descriptor, serviceName)) {
176*ee3b7b62SAndroid Build Coastguard Worker         tokenManager = new TokenManager();
177*ee3b7b62SAndroid Build Coastguard Worker         if (!manager->add(serviceName, tokenManager).withDefault(false)) {
178*ee3b7b62SAndroid Build Coastguard Worker             ALOGE("Failed to register ITokenManager with hwservicemanager.");
179*ee3b7b62SAndroid Build Coastguard Worker         }
180*ee3b7b62SAndroid Build Coastguard Worker     } else {
181*ee3b7b62SAndroid Build Coastguard Worker         ALOGW("Not registering android.hidl.token service because it is no longer supported");
182*ee3b7b62SAndroid Build Coastguard Worker     }
183*ee3b7b62SAndroid Build Coastguard Worker 
184*ee3b7b62SAndroid Build Coastguard Worker     // Tell IPCThreadState we're the service manager
185*ee3b7b62SAndroid Build Coastguard Worker     sp<IBinder> binder = toBinder<IServiceManager>(manager);
186*ee3b7b62SAndroid Build Coastguard Worker     sp<BHwBinder> service = static_cast<BHwBinder*>(binder.get()); // local binder object
187*ee3b7b62SAndroid Build Coastguard Worker     IPCThreadState::self()->setTheContextObject(service);
188*ee3b7b62SAndroid Build Coastguard Worker     // Then tell the kernel
189*ee3b7b62SAndroid Build Coastguard Worker     ProcessState::self()->becomeContextManager();
190*ee3b7b62SAndroid Build Coastguard Worker 
191*ee3b7b62SAndroid Build Coastguard Worker     int rc = property_set("hwservicemanager.ready", "true");
192*ee3b7b62SAndroid Build Coastguard Worker     if (rc) {
193*ee3b7b62SAndroid Build Coastguard Worker         ALOGE("Failed to set \"hwservicemanager.ready\" (error %d). "\
194*ee3b7b62SAndroid Build Coastguard Worker               "HAL services will not start!\n", rc);
195*ee3b7b62SAndroid Build Coastguard Worker     }
196*ee3b7b62SAndroid Build Coastguard Worker 
197*ee3b7b62SAndroid Build Coastguard Worker     sp<Looper> looper = Looper::prepare(0 /* opts */);
198*ee3b7b62SAndroid Build Coastguard Worker 
199*ee3b7b62SAndroid Build Coastguard Worker     (void)HwBinderCallback::setupTo(looper);
200*ee3b7b62SAndroid Build Coastguard Worker     (void)ClientCallbackCallback::setupTo(looper, manager);
201*ee3b7b62SAndroid Build Coastguard Worker 
202*ee3b7b62SAndroid Build Coastguard Worker     ALOGI("hwservicemanager is ready now.");
203*ee3b7b62SAndroid Build Coastguard Worker 
204*ee3b7b62SAndroid Build Coastguard Worker     while (true) {
205*ee3b7b62SAndroid Build Coastguard Worker         looper->pollAll(-1 /* timeoutMillis */);
206*ee3b7b62SAndroid Build Coastguard Worker     }
207*ee3b7b62SAndroid Build Coastguard Worker 
208*ee3b7b62SAndroid Build Coastguard Worker     return 0;
209*ee3b7b62SAndroid Build Coastguard Worker }
210