blob: dae431057209fe0b5061f539bdc1d28503a65d21 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright 2008, 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 */
Mike Lockwood270e87f2009-05-22 10:02:14 -040016#define LOG_TAG "SensorManager"
17
Peng Xud5ec5f92016-04-27 14:49:25 -070018#include "JNIHelp.h"
19#include "android_os_MessageQueue.h"
20#include "core_jni_helpers.h"
21#include "jni.h"
destradaa9ba7c1c2015-02-10 15:04:43 -080022
Aravind Akella88445992015-02-26 17:05:28 -080023#include <ScopedUtfChars.h>
Jeff Brown3b4049e2015-04-17 15:22:27 -070024#include <ScopedLocalRef.h>
Peng Xud5ec5f92016-04-27 14:49:25 -070025#include <android_runtime/AndroidRuntime.h>
Peng Xu7d96fa02017-03-04 16:15:45 -080026#include <android_runtime/android_hardware_HardwareBuffer.h>
Mathias Agopiancaf89302017-03-06 16:06:49 -080027#include <sensor/Sensor.h>
28#include <sensor/SensorEventQueue.h>
29#include <sensor/SensorManager.h>
Peng Xufa2672b2016-12-07 03:54:44 -080030#include <cutils/native_handle.h>
Mathias Agopiandb772d82013-01-31 19:31:12 -080031#include <utils/Log.h>
32#include <utils/Looper.h>
Peng Xua35b5532016-01-20 00:05:45 -080033#include <utils/Vector.h>
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080034
Peng Xud5ec5f92016-04-27 14:49:25 -070035#include <map>
Andreas Gampe987f79f2014-11-18 17:29:46 -080036
Peng Xu4a4baef2016-03-16 16:56:39 -070037namespace {
38
39using namespace android;
40
41struct {
Mathias Agopiandb772d82013-01-31 19:31:12 -080042 jclass clazz;
43 jmethodID dispatchSensorEvent;
Aravind Akellab4c76b12013-06-27 12:04:16 -070044 jmethodID dispatchFlushCompleteEvent;
Peng Xua2958352016-01-26 18:42:17 -080045 jmethodID dispatchAdditionalInfoEvent;
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -080046} gBaseEventQueueClassInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080047
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080048struct SensorOffsets
49{
Peng Xua35b5532016-01-20 00:05:45 -080050 jclass clazz;
Peng Xub85b1fe2016-04-20 21:28:33 -070051 //fields
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080052 jfieldID name;
53 jfieldID vendor;
54 jfieldID version;
55 jfieldID handle;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080056 jfieldID range;
57 jfieldID resolution;
58 jfieldID power;
Mathias Agopian050b5622010-07-29 16:51:38 -070059 jfieldID minDelay;
Aravind Akellab4c76b12013-06-27 12:04:16 -070060 jfieldID fifoReservedEventCount;
61 jfieldID fifoMaxEventCount;
Aravind Akellacd9a7bb2014-04-07 22:55:21 +000062 jfieldID stringType;
63 jfieldID requiredPermission;
Aravind Akella2f8b9142014-05-12 14:39:29 -070064 jfieldID maxDelay;
Aravind Akella27900352014-06-03 19:20:42 -070065 jfieldID flags;
Peng Xub85b1fe2016-04-20 21:28:33 -070066 //methods
destradaa9ba7c1c2015-02-10 15:04:43 -080067 jmethodID setType;
Peng Xub85b1fe2016-04-20 21:28:33 -070068 jmethodID setUuid;
Peng Xua35b5532016-01-20 00:05:45 -080069 jmethodID init;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080070} gSensorOffsets;
71
Peng Xua35b5532016-01-20 00:05:45 -080072struct ListOffsets {
73 jclass clazz;
74 jmethodID add;
75} gListOffsets;
Mathias Agopiandb772d82013-01-31 19:31:12 -080076
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080077/*
Peng Xu4a4baef2016-03-16 16:56:39 -070078 * nativeClassInit is not inteneded to be thread-safe. It should be called before other native...
79 * functions (except nativeCreate).
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080080 */
Mathias Agopiandb772d82013-01-31 19:31:12 -080081static void
82nativeClassInit (JNIEnv *_env, jclass _this)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080083{
Peng Xua35b5532016-01-20 00:05:45 -080084 //android.hardware.Sensor
Mathias Agopiandb772d82013-01-31 19:31:12 -080085 SensorOffsets& sensorOffsets = gSensorOffsets;
Peng Xua35b5532016-01-20 00:05:45 -080086 jclass sensorClass = (jclass) _env->NewGlobalRef(_env->FindClass("android/hardware/Sensor"));
87 sensorOffsets.clazz = sensorClass;
Mathias Agopiandb772d82013-01-31 19:31:12 -080088 sensorOffsets.name = _env->GetFieldID(sensorClass, "mName", "Ljava/lang/String;");
89 sensorOffsets.vendor = _env->GetFieldID(sensorClass, "mVendor", "Ljava/lang/String;");
90 sensorOffsets.version = _env->GetFieldID(sensorClass, "mVersion", "I");
91 sensorOffsets.handle = _env->GetFieldID(sensorClass, "mHandle", "I");
Mathias Agopiandb772d82013-01-31 19:31:12 -080092 sensorOffsets.range = _env->GetFieldID(sensorClass, "mMaxRange", "F");
93 sensorOffsets.resolution = _env->GetFieldID(sensorClass, "mResolution","F");
94 sensorOffsets.power = _env->GetFieldID(sensorClass, "mPower", "F");
95 sensorOffsets.minDelay = _env->GetFieldID(sensorClass, "mMinDelay", "I");
Aravind Akellab4c76b12013-06-27 12:04:16 -070096 sensorOffsets.fifoReservedEventCount =
97 _env->GetFieldID(sensorClass, "mFifoReservedEventCount", "I");
98 sensorOffsets.fifoMaxEventCount = _env->GetFieldID(sensorClass, "mFifoMaxEventCount", "I");
Aravind Akellacd9a7bb2014-04-07 22:55:21 +000099 sensorOffsets.stringType = _env->GetFieldID(sensorClass, "mStringType", "Ljava/lang/String;");
100 sensorOffsets.requiredPermission = _env->GetFieldID(sensorClass, "mRequiredPermission",
101 "Ljava/lang/String;");
Aravind Akella2f8b9142014-05-12 14:39:29 -0700102 sensorOffsets.maxDelay = _env->GetFieldID(sensorClass, "mMaxDelay", "I");
Aravind Akella27900352014-06-03 19:20:42 -0700103 sensorOffsets.flags = _env->GetFieldID(sensorClass, "mFlags", "I");
Peng Xua35b5532016-01-20 00:05:45 -0800104
destradaa9ba7c1c2015-02-10 15:04:43 -0800105 sensorOffsets.setType = _env->GetMethodID(sensorClass, "setType", "(I)Z");
Peng Xub85b1fe2016-04-20 21:28:33 -0700106 sensorOffsets.setUuid = _env->GetMethodID(sensorClass, "setUuid", "(JJ)V");
Peng Xua35b5532016-01-20 00:05:45 -0800107 sensorOffsets.init = _env->GetMethodID(sensorClass, "<init>", "()V");
108
109 // java.util.List;
110 ListOffsets& listOffsets = gListOffsets;
111 jclass listClass = (jclass) _env->NewGlobalRef(_env->FindClass("java/util/List"));
112 listOffsets.clazz = listClass;
113 listOffsets.add = _env->GetMethodID(listClass, "add", "(Ljava/lang/Object;)Z");
destradaa9ba7c1c2015-02-10 15:04:43 -0800114}
115
116/**
117 * A key comparator predicate.
118 * It is used to intern strings associated with Sensor data.
119 * It defines a 'Strict weak ordering' for the interned strings.
120 */
121class InternedStringCompare {
122public:
123 bool operator()(const String8* string1, const String8* string2) const {
124 if (string1 == NULL) {
125 return string2 != NULL;
126 }
Aravind Akella88445992015-02-26 17:05:28 -0800127 if (string2 == NULL) {
128 return false;
129 }
destradaa9ba7c1c2015-02-10 15:04:43 -0800130 return string1->compare(*string2) < 0;
131 }
132};
133
134/**
135 * A localized interning mechanism for Sensor strings.
136 * We implement our own interning to avoid the overhead of using java.lang.String#intern().
137 * It is common that Vendor, StringType, and RequirePermission data is common between many of the
138 * Sensors, by interning the memory usage to represent Sensors is optimized.
139 */
140static jstring
141getInternedString(JNIEnv *env, const String8* string) {
142 static std::map<const String8*, jstring, InternedStringCompare> internedStrings;
143
144 jstring internedString;
145 std::map<const String8*, jstring>::iterator iterator = internedStrings.find(string);
146 if (iterator != internedStrings.end()) {
147 internedString = iterator->second;
148 } else {
149 jstring localString = env->NewStringUTF(string->string());
150 // we are implementing our own interning so expect these strings to be backed by global refs
151 internedString = (jstring) env->NewGlobalRef(localString);
152 internedStrings.insert(std::make_pair(string, internedString));
153 env->DeleteLocalRef(localString);
154 }
destradaa9ba7c1c2015-02-10 15:04:43 -0800155 return internedString;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800156}
157
Svet Ganovb9d71a62015-04-30 10:38:13 -0700158static jlong
159nativeCreate
160(JNIEnv *env, jclass clazz, jstring opPackageName)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800161{
Svet Ganovb9d71a62015-04-30 10:38:13 -0700162 ScopedUtfChars opPackageNameUtf(env, opPackageName);
Aravind Akella2af91d02015-07-29 18:39:52 -0700163 return (jlong) &SensorManager::getInstanceForPackage(String16(opPackageNameUtf.c_str()));
Svet Ganovb9d71a62015-04-30 10:38:13 -0700164}
165
Peng Xua35b5532016-01-20 00:05:45 -0800166static jobject
167translateNativeSensorToJavaSensor(JNIEnv *env, jobject sensor, const Sensor& nativeSensor) {
168 const SensorOffsets& sensorOffsets(gSensorOffsets);
169
170 if (sensor == NULL) {
171 // Sensor sensor = new Sensor();
172 sensor = env->NewObject(sensorOffsets.clazz, sensorOffsets.init, "");
173 }
174
175 if (sensor != NULL) {
176 jstring name = env->NewStringUTF(nativeSensor.getName().string());
177 jstring vendor = env->NewStringUTF(nativeSensor.getVendor().string());
178 jstring requiredPermission =
179 env->NewStringUTF(nativeSensor.getRequiredPermission().string());
180
181 env->SetObjectField(sensor, sensorOffsets.name, name);
182 env->SetObjectField(sensor, sensorOffsets.vendor, vendor);
183 env->SetIntField(sensor, sensorOffsets.version, nativeSensor.getVersion());
184 env->SetIntField(sensor, sensorOffsets.handle, nativeSensor.getHandle());
185 env->SetFloatField(sensor, sensorOffsets.range, nativeSensor.getMaxValue());
186 env->SetFloatField(sensor, sensorOffsets.resolution, nativeSensor.getResolution());
187 env->SetFloatField(sensor, sensorOffsets.power, nativeSensor.getPowerUsage());
188 env->SetIntField(sensor, sensorOffsets.minDelay, nativeSensor.getMinDelay());
189 env->SetIntField(sensor, sensorOffsets.fifoReservedEventCount,
190 nativeSensor.getFifoReservedEventCount());
191 env->SetIntField(sensor, sensorOffsets.fifoMaxEventCount,
192 nativeSensor.getFifoMaxEventCount());
193 env->SetObjectField(sensor, sensorOffsets.requiredPermission,
194 requiredPermission);
195 env->SetIntField(sensor, sensorOffsets.maxDelay, nativeSensor.getMaxDelay());
196 env->SetIntField(sensor, sensorOffsets.flags, nativeSensor.getFlags());
Peng Xub85b1fe2016-04-20 21:28:33 -0700197
Peng Xua35b5532016-01-20 00:05:45 -0800198 if (env->CallBooleanMethod(sensor, sensorOffsets.setType, nativeSensor.getType())
199 == JNI_FALSE) {
200 jstring stringType = getInternedString(env, &nativeSensor.getStringType());
201 env->SetObjectField(sensor, sensorOffsets.stringType, stringType);
202 }
Peng Xub85b1fe2016-04-20 21:28:33 -0700203
Peng Xu716866d2016-05-25 21:10:00 -0700204 // TODO(b/29547335): Rename "setUuid" method to "setId".
205 int64_t id = nativeSensor.getId();
206 env->CallVoidMethod(sensor, sensorOffsets.setUuid, id, 0);
Peng Xua35b5532016-01-20 00:05:45 -0800207 }
208 return sensor;
209}
210
Aravind Akella516e40e2015-07-01 16:54:24 -0700211static jboolean
212nativeGetSensorAtIndex(JNIEnv *env, jclass clazz, jlong sensorManager, jobject sensor, jint index)
Svet Ganovb9d71a62015-04-30 10:38:13 -0700213{
214 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800215
Mathias Agopian1bf79782010-07-14 23:41:37 -0700216 Sensor const* const* sensorList;
Peng Xua08c9f52016-02-04 22:27:20 -0800217 ssize_t count = mgr->getSensorList(&sensorList);
218 if (ssize_t(index) >= count) {
Aravind Akella516e40e2015-07-01 16:54:24 -0700219 return false;
destradaa9ba7c1c2015-02-10 15:04:43 -0800220 }
Aravind Akellab4c76b12013-06-27 12:04:16 -0700221
Peng Xub85b1fe2016-04-20 21:28:33 -0700222 return translateNativeSensorToJavaSensor(env, sensor, *sensorList[index]) != NULL;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800223}
224
Peng Xua35b5532016-01-20 00:05:45 -0800225static void
226nativeGetDynamicSensors(JNIEnv *env, jclass clazz, jlong sensorManager, jobject sensorList) {
227
228 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
229 const ListOffsets& listOffsets(gListOffsets);
230
231 Vector<Sensor> nativeList;
232
233 mgr->getDynamicSensorList(nativeList);
234
Ben Wagner85ea32b2016-03-16 17:10:42 -0400235 ALOGI("DYNS native SensorManager.getDynamicSensorList return %zu sensors", nativeList.size());
Peng Xua35b5532016-01-20 00:05:45 -0800236 for (size_t i = 0; i < nativeList.size(); ++i) {
237 jobject sensor = translateNativeSensorToJavaSensor(env, NULL, nativeList[i]);
238 // add to list
239 env->CallBooleanMethod(sensorList, listOffsets.add, sensor);
240 }
241}
242
Aravind Akellad123b512015-06-29 12:35:51 -0700243static jboolean nativeIsDataInjectionEnabled(JNIEnv *_env, jclass _this, jlong sensorManager) {
Svet Ganovb9d71a62015-04-30 10:38:13 -0700244 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
Aravind Akellad123b512015-06-29 12:35:51 -0700245 return mgr->isDataInjectionEnabled();
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700246}
247
Peng Xufa2672b2016-12-07 03:54:44 -0800248static jint nativeCreateDirectChannel(JNIEnv *_env, jclass _this, jlong sensorManager,
Peng Xu7d96fa02017-03-04 16:15:45 -0800249 jlong size, jint channelType, jint fd, jobject hardwareBufferObj) {
250 const native_handle_t *nativeHandle = nullptr;
251 NATIVE_HANDLE_DECLARE_STORAGE(ashmemHandle, 1, 0);
Peng Xufa2672b2016-12-07 03:54:44 -0800252
Peng Xu7d96fa02017-03-04 16:15:45 -0800253 if (channelType == SENSOR_DIRECT_MEM_TYPE_ASHMEM) {
254 native_handle_t *handle = native_handle_init(ashmemHandle, 1, 0);
255 handle->data[0] = fd;
256 nativeHandle = handle;
257 } else if (channelType == SENSOR_DIRECT_MEM_TYPE_GRALLOC) {
258 AHardwareBuffer *hardwareBuffer =
259 android_hardware_HardwareBuffer_getNativeHardwareBuffer(_env, hardwareBufferObj);
260 if (hardwareBuffer != nullptr) {
261 nativeHandle = AHardwareBuffer_getNativeHandle(hardwareBuffer);
Peng Xufa2672b2016-12-07 03:54:44 -0800262 }
263 }
Peng Xu7d96fa02017-03-04 16:15:45 -0800264
265 if (nativeHandle == nullptr) {
266 return BAD_VALUE;
267 }
268
269 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
270 return mgr->createDirectChannel(size, channelType, nativeHandle);
Peng Xufa2672b2016-12-07 03:54:44 -0800271}
272
273static void nativeDestroyDirectChannel(JNIEnv *_env, jclass _this, jlong sensorManager,
274 jint channelHandle) {
275 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
276 mgr->destroyDirectChannel(channelHandle);
277}
278
279static jint nativeConfigDirectChannel(JNIEnv *_env, jclass _this, jlong sensorManager,
280 jint channelHandle, jint sensorHandle, jint rate) {
281 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
282 return mgr->configureDirectChannel(channelHandle, sensorHandle, rate);
283}
284
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800285//----------------------------------------------------------------------------
Mathias Agopiandb772d82013-01-31 19:31:12 -0800286
287class Receiver : public LooperCallback {
288 sp<SensorEventQueue> mSensorQueue;
289 sp<MessageQueue> mMessageQueue;
Jeff Brown3b4049e2015-04-17 15:22:27 -0700290 jobject mReceiverWeakGlobal;
Peng Xua2958352016-01-26 18:42:17 -0800291 jfloatArray mFloatScratch;
292 jintArray mIntScratch;
Mathias Agopiandb772d82013-01-31 19:31:12 -0800293public:
294 Receiver(const sp<SensorEventQueue>& sensorQueue,
295 const sp<MessageQueue>& messageQueue,
Peng Xua2958352016-01-26 18:42:17 -0800296 jobject receiverWeak) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800297 JNIEnv* env = AndroidRuntime::getJNIEnv();
298 mSensorQueue = sensorQueue;
299 mMessageQueue = messageQueue;
Jeff Brown3b4049e2015-04-17 15:22:27 -0700300 mReceiverWeakGlobal = env->NewGlobalRef(receiverWeak);
Peng Xua2958352016-01-26 18:42:17 -0800301
302 mIntScratch = (jintArray) env->NewGlobalRef(env->NewIntArray(16));
303 mFloatScratch = (jfloatArray) env->NewGlobalRef(env->NewFloatArray(16));
Mathias Agopiandb772d82013-01-31 19:31:12 -0800304 }
305 ~Receiver() {
306 JNIEnv* env = AndroidRuntime::getJNIEnv();
Jeff Brown3b4049e2015-04-17 15:22:27 -0700307 env->DeleteGlobalRef(mReceiverWeakGlobal);
Peng Xua2958352016-01-26 18:42:17 -0800308 env->DeleteGlobalRef(mFloatScratch);
309 env->DeleteGlobalRef(mIntScratch);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800310 }
311 sp<SensorEventQueue> getSensorEventQueue() const {
312 return mSensorQueue;
313 }
314
315 void destroy() {
316 mMessageQueue->getLooper()->removeFd( mSensorQueue->getFd() );
317 }
318
319private:
320 virtual void onFirstRef() {
321 LooperCallback::onFirstRef();
322 mMessageQueue->getLooper()->addFd(mSensorQueue->getFd(), 0,
323 ALOOPER_EVENT_INPUT, this, mSensorQueue.get());
324 }
325
326 virtual int handleEvent(int fd, int events, void* data) {
327 JNIEnv* env = AndroidRuntime::getJNIEnv();
328 sp<SensorEventQueue> q = reinterpret_cast<SensorEventQueue *>(data);
Jeff Brown3b4049e2015-04-17 15:22:27 -0700329 ScopedLocalRef<jobject> receiverObj(env, jniGetReferent(env, mReceiverWeakGlobal));
330
Mathias Agopiandb772d82013-01-31 19:31:12 -0800331 ssize_t n;
332 ASensorEvent buffer[16];
333 while ((n = q->read(buffer, 16)) > 0) {
334 for (int i=0 ; i<n ; i++) {
Aravind Akella0288ca62014-07-28 14:53:44 -0700335 if (buffer[i].type == SENSOR_TYPE_STEP_COUNTER) {
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700336 // step-counter returns a uint64, but the java API only deals with floats
337 float value = float(buffer[i].u64.step_counter);
Peng Xua2958352016-01-26 18:42:17 -0800338 env->SetFloatArrayRegion(mFloatScratch, 0, 1, &value);
Peng Xua35b5532016-01-20 00:05:45 -0800339 } else if (buffer[i].type == SENSOR_TYPE_DYNAMIC_SENSOR_META) {
340 float value[2];
341 value[0] = buffer[i].dynamic_sensor_meta.connected ? 1.f: 0.f;
342 value[1] = float(buffer[i].dynamic_sensor_meta.handle);
Peng Xua2958352016-01-26 18:42:17 -0800343 env->SetFloatArrayRegion(mFloatScratch, 0, 2, value);
344 } else if (buffer[i].type == SENSOR_TYPE_ADDITIONAL_INFO) {
345 env->SetIntArrayRegion(mIntScratch, 0, 14,
346 buffer[i].additional_info.data_int32);
347 env->SetFloatArrayRegion(mFloatScratch, 0, 14,
348 buffer[i].additional_info.data_float);
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700349 } else {
Peng Xua2958352016-01-26 18:42:17 -0800350 env->SetFloatArrayRegion(mFloatScratch, 0, 16, buffer[i].data);
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700351 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800352
Aravind Akellab4c76b12013-06-27 12:04:16 -0700353 if (buffer[i].type == SENSOR_TYPE_META_DATA) {
354 // This is a flush complete sensor event. Call dispatchFlushCompleteEvent
355 // method.
Jeff Brown3b4049e2015-04-17 15:22:27 -0700356 if (receiverObj.get()) {
357 env->CallVoidMethod(receiverObj.get(),
358 gBaseEventQueueClassInfo.dispatchFlushCompleteEvent,
359 buffer[i].meta_data.sensor);
360 }
Peng Xua2958352016-01-26 18:42:17 -0800361 } else if (buffer[i].type == SENSOR_TYPE_ADDITIONAL_INFO) {
362 // This is a flush complete sensor event. Call dispatchAdditionalInfoEvent
363 // method.
364 if (receiverObj.get()) {
365 int type = buffer[i].additional_info.type;
366 int serial = buffer[i].additional_info.serial;
367 env->CallVoidMethod(receiverObj.get(),
368 gBaseEventQueueClassInfo.dispatchAdditionalInfoEvent,
369 buffer[i].sensor,
370 type, serial,
371 mFloatScratch,
372 mIntScratch,
373 buffer[i].timestamp);
374 }
375 }else {
Etienne Le Grandaf805102014-05-16 12:21:41 -0700376 int8_t status;
377 switch (buffer[i].type) {
378 case SENSOR_TYPE_ORIENTATION:
379 case SENSOR_TYPE_MAGNETIC_FIELD:
380 case SENSOR_TYPE_ACCELEROMETER:
381 case SENSOR_TYPE_GYROSCOPE:
Aravind Akellaf38226a2015-06-23 10:32:16 -0700382 case SENSOR_TYPE_GRAVITY:
383 case SENSOR_TYPE_LINEAR_ACCELERATION:
Etienne Le Grandaf805102014-05-16 12:21:41 -0700384 status = buffer[i].vector.status;
385 break;
386 case SENSOR_TYPE_HEART_RATE:
387 status = buffer[i].heart_rate.status;
388 break;
389 default:
390 status = SENSOR_STATUS_ACCURACY_HIGH;
391 break;
392 }
Jeff Brown3b4049e2015-04-17 15:22:27 -0700393 if (receiverObj.get()) {
394 env->CallVoidMethod(receiverObj.get(),
395 gBaseEventQueueClassInfo.dispatchSensorEvent,
396 buffer[i].sensor,
Peng Xua2958352016-01-26 18:42:17 -0800397 mFloatScratch,
Jeff Brown3b4049e2015-04-17 15:22:27 -0700398 status,
399 buffer[i].timestamp);
400 }
Aravind Akellab4c76b12013-06-27 12:04:16 -0700401 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800402 if (env->ExceptionCheck()) {
Aravind Akella35187bd2014-02-11 18:44:42 -0800403 mSensorQueue->sendAck(buffer, n);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800404 ALOGE("Exception dispatching input event.");
405 return 1;
406 }
407 }
Aravind Akella35187bd2014-02-11 18:44:42 -0800408 mSensorQueue->sendAck(buffer, n);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800409 }
410 if (n<0 && n != -EAGAIN) {
411 // FIXME: error receiving events, what to do in this case?
412 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800413 return 1;
414 }
415};
416
Svet Ganovb9d71a62015-04-30 10:38:13 -0700417static jlong nativeInitSensorEventQueue(JNIEnv *env, jclass clazz, jlong sensorManager,
Peng Xua2958352016-01-26 18:42:17 -0800418 jobject eventQWeak, jobject msgQ, jstring packageName, jint mode) {
Svet Ganovb9d71a62015-04-30 10:38:13 -0700419 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
Aravind Akella88445992015-02-26 17:05:28 -0800420 ScopedUtfChars packageUtf(env, packageName);
421 String8 clientName(packageUtf.c_str());
Svet Ganovb9d71a62015-04-30 10:38:13 -0700422 sp<SensorEventQueue> queue(mgr->createEventQueue(clientName, mode));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800423
Mathias Agopiandb772d82013-01-31 19:31:12 -0800424 sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, msgQ);
425 if (messageQueue == NULL) {
426 jniThrowRuntimeException(env, "MessageQueue is not initialized.");
427 return 0;
Mathias Agopian0b6d77b2012-05-07 18:27:06 -0700428 }
Mathias Agopian1bf79782010-07-14 23:41:37 -0700429
Peng Xua2958352016-01-26 18:42:17 -0800430 sp<Receiver> receiver = new Receiver(queue, messageQueue, eventQWeak);
Mathias Agopianb1d90c82013-03-06 17:45:42 -0800431 receiver->incStrong((void*)nativeInitSensorEventQueue);
Ashok Bhat4838e332014-01-03 14:37:19 +0000432 return jlong(receiver.get());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800433}
434
Ashok Bhat4838e332014-01-03 14:37:19 +0000435static jint nativeEnableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle, jint rate_us,
Aravind Akella88445992015-02-26 17:05:28 -0800436 jint maxBatchReportLatency) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800437 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
Aravind Akellab4c76b12013-06-27 12:04:16 -0700438 return receiver->getSensorEventQueue()->enableSensor(handle, rate_us, maxBatchReportLatency,
Aravind Akella88445992015-02-26 17:05:28 -0800439 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800440}
441
Ashok Bhat4838e332014-01-03 14:37:19 +0000442static jint nativeDisableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800443 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
444 return receiver->getSensorEventQueue()->disableSensor(handle);
445}
Mathias Agopian1bf79782010-07-14 23:41:37 -0700446
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700447static void nativeDestroySensorEventQueue(JNIEnv *env, jclass clazz, jlong eventQ) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800448 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
449 receiver->destroy();
Mathias Agopianb1d90c82013-03-06 17:45:42 -0800450 receiver->decStrong((void*)nativeInitSensorEventQueue);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800451}
Mathias Agopian1bf79782010-07-14 23:41:37 -0700452
Ashok Bhat4838e332014-01-03 14:37:19 +0000453static jint nativeFlushSensor(JNIEnv *env, jclass clazz, jlong eventQ) {
Aravind Akellab4c76b12013-06-27 12:04:16 -0700454 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
Aravind Akella4bdc37d2013-10-01 17:58:35 -0700455 return receiver->getSensorEventQueue()->flush();
Aravind Akellab4c76b12013-06-27 12:04:16 -0700456}
Mathias Agopiandb772d82013-01-31 19:31:12 -0800457
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700458static jint nativeInjectSensorData(JNIEnv *env, jclass clazz, jlong eventQ, jint handle,
459 jfloatArray values, jint accuracy, jlong timestamp) {
460 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
461 // Create a sensor_event from the above data which can be injected into the HAL.
462 ASensorEvent sensor_event;
463 memset(&sensor_event, 0, sizeof(sensor_event));
464 sensor_event.sensor = handle;
465 sensor_event.timestamp = timestamp;
466 env->GetFloatArrayRegion(values, 0, env->GetArrayLength(values), sensor_event.data);
467 return receiver->getSensorEventQueue()->injectSensorEvent(sensor_event);
468}
Mathias Agopiandb772d82013-01-31 19:31:12 -0800469//----------------------------------------------------------------------------
470
Daniel Micay76f6a862015-09-19 17:31:01 -0400471static const JNINativeMethod gSystemSensorManagerMethods[] = {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800472 {"nativeClassInit",
473 "()V",
474 (void*)nativeClassInit },
Svet Ganovb9d71a62015-04-30 10:38:13 -0700475 {"nativeCreate",
476 "(Ljava/lang/String;)J",
477 (void*)nativeCreate },
Mathias Agopiandb772d82013-01-31 19:31:12 -0800478
Aravind Akella516e40e2015-07-01 16:54:24 -0700479 {"nativeGetSensorAtIndex",
480 "(JLandroid/hardware/Sensor;I)Z",
481 (void*)nativeGetSensorAtIndex },
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700482
Peng Xua35b5532016-01-20 00:05:45 -0800483 {"nativeGetDynamicSensors",
484 "(JLjava/util/List;)V",
485 (void*)nativeGetDynamicSensors },
486
Aravind Akellad123b512015-06-29 12:35:51 -0700487 {"nativeIsDataInjectionEnabled",
488 "(J)Z",
Peng Xufa2672b2016-12-07 03:54:44 -0800489 (void*)nativeIsDataInjectionEnabled },
490
491 {"nativeCreateDirectChannel",
Peng Xu7d96fa02017-03-04 16:15:45 -0800492 "(JJIILandroid/hardware/HardwareBuffer;)I",
Peng Xufa2672b2016-12-07 03:54:44 -0800493 (void*)nativeCreateDirectChannel },
494
495 {"nativeDestroyDirectChannel",
496 "(JI)V",
497 (void*)nativeDestroyDirectChannel },
498
499 {"nativeConfigDirectChannel",
500 "(JIII)I",
501 (void*)nativeConfigDirectChannel },
Mathias Agopiandb772d82013-01-31 19:31:12 -0800502};
503
Daniel Micay76f6a862015-09-19 17:31:01 -0400504static const JNINativeMethod gBaseEventQueueMethods[] = {
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -0800505 {"nativeInitBaseEventQueue",
Peng Xua2958352016-01-26 18:42:17 -0800506 "(JLjava/lang/ref/WeakReference;Landroid/os/MessageQueue;Ljava/lang/String;ILjava/lang/String;)J",
Svet Ganovb9d71a62015-04-30 10:38:13 -0700507 (void*)nativeInitSensorEventQueue },
Mathias Agopiandb772d82013-01-31 19:31:12 -0800508
509 {"nativeEnableSensor",
Aravind Akella88445992015-02-26 17:05:28 -0800510 "(JIII)I",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800511 (void*)nativeEnableSensor },
512
513 {"nativeDisableSensor",
Ashok Bhat4838e332014-01-03 14:37:19 +0000514 "(JI)I",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800515 (void*)nativeDisableSensor },
516
517 {"nativeDestroySensorEventQueue",
Ashok Bhat4838e332014-01-03 14:37:19 +0000518 "(J)V",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800519 (void*)nativeDestroySensorEventQueue },
Aravind Akellab4c76b12013-06-27 12:04:16 -0700520
521 {"nativeFlushSensor",
Ashok Bhat4838e332014-01-03 14:37:19 +0000522 "(J)I",
Aravind Akellab4c76b12013-06-27 12:04:16 -0700523 (void*)nativeFlushSensor },
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700524
525 {"nativeInjectSensorData",
526 "(JI[FIJ)I",
527 (void*)nativeInjectSensorData },
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800528};
529
Peng Xu4a4baef2016-03-16 16:56:39 -0700530} //unnamed namespace
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800531
532int register_android_hardware_SensorManager(JNIEnv *env)
533{
Andreas Gampe987f79f2014-11-18 17:29:46 -0800534 RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800535 gSystemSensorManagerMethods, NELEM(gSystemSensorManagerMethods));
536
Andreas Gampe987f79f2014-11-18 17:29:46 -0800537 RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager$BaseEventQueue",
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -0800538 gBaseEventQueueMethods, NELEM(gBaseEventQueueMethods));
Mathias Agopiandb772d82013-01-31 19:31:12 -0800539
Andreas Gampe987f79f2014-11-18 17:29:46 -0800540 gBaseEventQueueClassInfo.clazz = FindClassOrDie(env,
541 "android/hardware/SystemSensorManager$BaseEventQueue");
Mathias Agopiandb772d82013-01-31 19:31:12 -0800542
Andreas Gampe987f79f2014-11-18 17:29:46 -0800543 gBaseEventQueueClassInfo.dispatchSensorEvent = GetMethodIDOrDie(env,
544 gBaseEventQueueClassInfo.clazz, "dispatchSensorEvent", "(I[FIJ)V");
Mathias Agopiandb772d82013-01-31 19:31:12 -0800545
Andreas Gampe987f79f2014-11-18 17:29:46 -0800546 gBaseEventQueueClassInfo.dispatchFlushCompleteEvent = GetMethodIDOrDie(env,
547 gBaseEventQueueClassInfo.clazz, "dispatchFlushCompleteEvent", "(I)V");
Aravind Akellab4c76b12013-06-27 12:04:16 -0700548
Peng Xua2958352016-01-26 18:42:17 -0800549 gBaseEventQueueClassInfo.dispatchAdditionalInfoEvent = GetMethodIDOrDie(env,
550 gBaseEventQueueClassInfo.clazz, "dispatchAdditionalInfoEvent", "(III[F[I)V");
551
Mathias Agopiandb772d82013-01-31 19:31:12 -0800552 return 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800553}