xref: /aosp_15_r20/frameworks/native/services/sensorservice/hidl/SensorManager.cpp (revision 38e8c45f13ce32b0dcecb25141ffecaf386fa17f)
1 /*
2  * Copyright (C) 2017 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 // LOG_TAG defined via build flag.
18 #ifndef LOG_TAG
19 #define LOG_TAG "HidlSensorManager"
20 #endif
21 #include <android-base/logging.h>
22 
23 #include "SensorManager.h"
24 
25 #include <sched.h>
26 
27 
28 #include "EventQueue.h"
29 #include "DirectReportChannel.h"
30 #include "utils.h"
31 
32 #include <hwbinder/IPCThreadState.h>
33 #include <utils/String8.h>
34 
35 namespace android {
36 namespace frameworks {
37 namespace sensorservice {
38 namespace V1_0 {
39 namespace implementation {
40 
41 using ::android::hardware::sensors::V1_0::SensorInfo;
42 using ::android::hardware::sensors::V1_0::SensorsEventFormatOffset;
43 using ::android::hardware::hidl_vec;
44 using ::android::hardware::Void;
45 
46 static const char* POLL_THREAD_NAME = "hidl_ssvc_poll";
47 
SensorManager(JavaVM * vm)48 SensorManager::SensorManager(JavaVM* vm)
49         : mLooper(new Looper(false /*allowNonCallbacks*/)), mStopThread(true), mJavaVm(vm) {
50 }
51 
~SensorManager()52 SensorManager::~SensorManager() {
53     // Stops pollAll inside the thread.
54     std::lock_guard<std::mutex> lock(mThreadMutex);
55 
56     mStopThread = true;
57     if (mLooper != nullptr) {
58         mLooper->wake();
59     }
60     if (mPollThread.joinable()) {
61         mPollThread.join();
62     }
63 
64     ::android::SensorManager::removeInstanceForPackage(
65             String16(ISensorManager::descriptor));
66 }
67 
68 // Methods from ::android::frameworks::sensorservice::V1_0::ISensorManager follow.
getSensorList(getSensorList_cb _hidl_cb)69 Return<void> SensorManager::getSensorList(getSensorList_cb _hidl_cb) {
70     ::android::Sensor const* const* list;
71     ssize_t count = getInternalManager().getSensorList(&list);
72     if (count < 0 || !list) {
73         LOG(ERROR) << "::android::SensorManager::getSensorList encounters " << count;
74         _hidl_cb({}, Result::UNKNOWN_ERROR);
75         return Void();
76     }
77     hidl_vec<SensorInfo> ret;
78     ret.resize(static_cast<size_t>(count));
79     for (ssize_t i = 0; i < count; ++i) {
80         ret[i] = convertSensor(*list[i]);
81     }
82     _hidl_cb(ret, Result::OK);
83     return Void();
84 }
85 
getDefaultSensor(SensorType type,getDefaultSensor_cb _hidl_cb)86 Return<void> SensorManager::getDefaultSensor(SensorType type, getDefaultSensor_cb _hidl_cb) {
87     ::android::Sensor const* sensor = getInternalManager().getDefaultSensor(static_cast<int>(type));
88     if (!sensor) {
89         _hidl_cb({}, Result::NOT_EXIST);
90         return Void();
91     }
92     _hidl_cb(convertSensor(*sensor), Result::OK);
93     return Void();
94 }
95 
96 template<typename Callback>
createDirectChannel(::android::SensorManager & manager,size_t size,int type,const native_handle_t * handle,const Callback & _hidl_cb)97 void createDirectChannel(::android::SensorManager& manager, size_t size, int type,
98         const native_handle_t* handle, const Callback& _hidl_cb) {
99 
100     int channelId = manager.createDirectChannel(
101         size, type, handle);
102     if (channelId < 0) {
103         _hidl_cb(nullptr, convertResult(channelId));
104         return;
105     }
106     if (channelId == 0) {
107         _hidl_cb(nullptr, Result::UNKNOWN_ERROR);
108         return;
109     }
110 
111     _hidl_cb(sp<IDirectReportChannel>(new DirectReportChannel(manager, channelId)),
112             Result::OK);
113 }
114 
createAshmemDirectChannel(const hidl_memory & mem,uint64_t size,createAshmemDirectChannel_cb _hidl_cb)115 Return<void> SensorManager::createAshmemDirectChannel(
116         const hidl_memory& mem, uint64_t size,
117         createAshmemDirectChannel_cb _hidl_cb) {
118     if (size > mem.size() || size < (uint64_t)SensorsEventFormatOffset::TOTAL_LENGTH) {
119         _hidl_cb(nullptr, Result::BAD_VALUE);
120         return Void();
121     }
122 
123     createDirectChannel(getInternalManager(), size, SENSOR_DIRECT_MEM_TYPE_ASHMEM,
124             mem.handle(), _hidl_cb);
125 
126     return Void();
127 }
128 
createGrallocDirectChannel(const hidl_handle & buffer,uint64_t size,createGrallocDirectChannel_cb _hidl_cb)129 Return<void> SensorManager::createGrallocDirectChannel(
130         const hidl_handle& buffer, uint64_t size,
131         createGrallocDirectChannel_cb _hidl_cb) {
132 
133     createDirectChannel(getInternalManager(), size, SENSOR_DIRECT_MEM_TYPE_GRALLOC,
134             buffer.getNativeHandle(), _hidl_cb);
135 
136     return Void();
137 }
138 
139 /* One global looper for all event queues created from this SensorManager. */
getLooper()140 sp<Looper> SensorManager::getLooper() {
141     std::lock_guard<std::mutex> lock(mThreadMutex);
142 
143     if (!mPollThread.joinable()) {
144         // if thread not initialized, start thread
145         mStopThread = false;
146         std::thread pollThread{[&stopThread = mStopThread, looper = mLooper, javaVm = mJavaVm] {
147 
148             struct sched_param p = {0};
149             p.sched_priority = 10;
150             if (sched_setscheduler(0 /* current thread*/, SCHED_FIFO, &p) != 0) {
151                 LOG(WARNING) << "Could not use SCHED_FIFO for looper thread: "
152                         << strerror(errno);
153             }
154 
155             // set looper
156             Looper::setForThread(looper);
157 
158             // Attach the thread to JavaVM so that pollAll do not crash if the thread
159             // eventually calls into Java.
160             JavaVMAttachArgs args{
161                 .version = JNI_VERSION_1_2,
162                 .name = POLL_THREAD_NAME,
163                 .group = nullptr
164             };
165             JNIEnv* env;
166             if (javaVm->AttachCurrentThread(&env, &args) != JNI_OK) {
167                 LOG(FATAL) << "Cannot attach SensorManager looper thread to Java VM.";
168             }
169 
170             LOG(INFO) << POLL_THREAD_NAME << " started.";
171             for (;;) {
172                 int pollResult = looper->pollAll(-1 /* timeout */);
173                 if (pollResult == Looper::POLL_WAKE) {
174                     if (stopThread == true) {
175                         LOG(INFO) << POLL_THREAD_NAME << ": requested to stop";
176                         break;
177                     } else {
178                         LOG(INFO) << POLL_THREAD_NAME << ": spurious wake up, back to work";
179                     }
180                 } else {
181                     LOG(ERROR) << POLL_THREAD_NAME << ": Looper::pollAll returns unexpected "
182                                << pollResult;
183                     break;
184                 }
185             }
186 
187             if (javaVm->DetachCurrentThread() != JNI_OK) {
188                 LOG(ERROR) << "Cannot detach SensorManager looper thread from Java VM.";
189             }
190 
191             LOG(INFO) << POLL_THREAD_NAME << " is terminated.";
192         }};
193         mPollThread = std::move(pollThread);
194     }
195     return mLooper;
196 }
197 
getInternalManager()198 ::android::SensorManager& SensorManager::getInternalManager() {
199     return ::android::SensorManager::getInstanceForPackage(
200             String16(ISensorManager::descriptor));
201 }
202 
createEventQueue(const sp<IEventQueueCallback> & callback,createEventQueue_cb _hidl_cb)203 Return<void> SensorManager::createEventQueue(
204         const sp<IEventQueueCallback> &callback, createEventQueue_cb _hidl_cb) {
205     if (callback == nullptr) {
206         _hidl_cb(nullptr, Result::BAD_VALUE);
207         return Void();
208     }
209 
210     sp<::android::Looper> looper = getLooper();
211     if (looper == nullptr) {
212         LOG(ERROR) << "::android::SensorManager::createEventQueue cannot initialize looper";
213         _hidl_cb(nullptr, Result::UNKNOWN_ERROR);
214         return Void();
215     }
216 
217     String8 package(String8::format("hidl_client_pid_%d",
218                                     android::hardware::IPCThreadState::self()->getCallingPid()));
219     sp<::android::SensorEventQueue> internalQueue = getInternalManager().createEventQueue(package);
220     if (internalQueue == nullptr) {
221         LOG(WARNING) << "::android::SensorManager::createEventQueue returns nullptr.";
222         _hidl_cb(nullptr, Result::UNKNOWN_ERROR);
223         return Void();
224     }
225 
226     sp<IEventQueue> queue = new EventQueue(callback, looper, internalQueue);
227     _hidl_cb(queue, Result::OK);
228 
229     return Void();
230 }
231 
232 }  // namespace implementation
233 }  // namespace V1_0
234 }  // namespace sensorservice
235 }  // namespace frameworks
236 }  // namespace android
237