blob: 271f24b1305249fcbb764123bfe1ab2c10e1dc0f [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>
26#include <gui/Sensor.h>
27#include <gui/SensorEventQueue.h>
28#include <gui/SensorManager.h>
Mathias Agopiandb772d82013-01-31 19:31:12 -080029#include <utils/Log.h>
30#include <utils/Looper.h>
Peng Xua35b5532016-01-20 00:05:45 -080031#include <utils/Vector.h>
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080032
Peng Xud5ec5f92016-04-27 14:49:25 -070033#include <map>
Andreas Gampe987f79f2014-11-18 17:29:46 -080034
Peng Xu4a4baef2016-03-16 16:56:39 -070035namespace {
36
37using namespace android;
38
39struct {
Mathias Agopiandb772d82013-01-31 19:31:12 -080040 jclass clazz;
41 jmethodID dispatchSensorEvent;
Aravind Akellab4c76b12013-06-27 12:04:16 -070042 jmethodID dispatchFlushCompleteEvent;
Peng Xua2958352016-01-26 18:42:17 -080043 jmethodID dispatchAdditionalInfoEvent;
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -080044} gBaseEventQueueClassInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080045
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080046struct SensorOffsets
47{
Peng Xua35b5532016-01-20 00:05:45 -080048 jclass clazz;
Peng Xub85b1fe2016-04-20 21:28:33 -070049 //fields
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080050 jfieldID name;
51 jfieldID vendor;
52 jfieldID version;
53 jfieldID handle;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080054 jfieldID range;
55 jfieldID resolution;
56 jfieldID power;
Mathias Agopian050b5622010-07-29 16:51:38 -070057 jfieldID minDelay;
Aravind Akellab4c76b12013-06-27 12:04:16 -070058 jfieldID fifoReservedEventCount;
59 jfieldID fifoMaxEventCount;
Aravind Akellacd9a7bb2014-04-07 22:55:21 +000060 jfieldID stringType;
61 jfieldID requiredPermission;
Aravind Akella2f8b9142014-05-12 14:39:29 -070062 jfieldID maxDelay;
Aravind Akella27900352014-06-03 19:20:42 -070063 jfieldID flags;
Peng Xub85b1fe2016-04-20 21:28:33 -070064 //methods
destradaa9ba7c1c2015-02-10 15:04:43 -080065 jmethodID setType;
Peng Xub85b1fe2016-04-20 21:28:33 -070066 jmethodID setUuid;
Peng Xua35b5532016-01-20 00:05:45 -080067 jmethodID init;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080068} gSensorOffsets;
69
Peng Xua35b5532016-01-20 00:05:45 -080070struct ListOffsets {
71 jclass clazz;
72 jmethodID add;
73} gListOffsets;
Mathias Agopiandb772d82013-01-31 19:31:12 -080074
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080075/*
Peng Xu4a4baef2016-03-16 16:56:39 -070076 * nativeClassInit is not inteneded to be thread-safe. It should be called before other native...
77 * functions (except nativeCreate).
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080078 */
Mathias Agopiandb772d82013-01-31 19:31:12 -080079static void
80nativeClassInit (JNIEnv *_env, jclass _this)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080081{
Peng Xua35b5532016-01-20 00:05:45 -080082 //android.hardware.Sensor
Mathias Agopiandb772d82013-01-31 19:31:12 -080083 SensorOffsets& sensorOffsets = gSensorOffsets;
Peng Xua35b5532016-01-20 00:05:45 -080084 jclass sensorClass = (jclass) _env->NewGlobalRef(_env->FindClass("android/hardware/Sensor"));
85 sensorOffsets.clazz = sensorClass;
Mathias Agopiandb772d82013-01-31 19:31:12 -080086 sensorOffsets.name = _env->GetFieldID(sensorClass, "mName", "Ljava/lang/String;");
87 sensorOffsets.vendor = _env->GetFieldID(sensorClass, "mVendor", "Ljava/lang/String;");
88 sensorOffsets.version = _env->GetFieldID(sensorClass, "mVersion", "I");
89 sensorOffsets.handle = _env->GetFieldID(sensorClass, "mHandle", "I");
Mathias Agopiandb772d82013-01-31 19:31:12 -080090 sensorOffsets.range = _env->GetFieldID(sensorClass, "mMaxRange", "F");
91 sensorOffsets.resolution = _env->GetFieldID(sensorClass, "mResolution","F");
92 sensorOffsets.power = _env->GetFieldID(sensorClass, "mPower", "F");
93 sensorOffsets.minDelay = _env->GetFieldID(sensorClass, "mMinDelay", "I");
Aravind Akellab4c76b12013-06-27 12:04:16 -070094 sensorOffsets.fifoReservedEventCount =
95 _env->GetFieldID(sensorClass, "mFifoReservedEventCount", "I");
96 sensorOffsets.fifoMaxEventCount = _env->GetFieldID(sensorClass, "mFifoMaxEventCount", "I");
Aravind Akellacd9a7bb2014-04-07 22:55:21 +000097 sensorOffsets.stringType = _env->GetFieldID(sensorClass, "mStringType", "Ljava/lang/String;");
98 sensorOffsets.requiredPermission = _env->GetFieldID(sensorClass, "mRequiredPermission",
99 "Ljava/lang/String;");
Aravind Akella2f8b9142014-05-12 14:39:29 -0700100 sensorOffsets.maxDelay = _env->GetFieldID(sensorClass, "mMaxDelay", "I");
Aravind Akella27900352014-06-03 19:20:42 -0700101 sensorOffsets.flags = _env->GetFieldID(sensorClass, "mFlags", "I");
Peng Xua35b5532016-01-20 00:05:45 -0800102
destradaa9ba7c1c2015-02-10 15:04:43 -0800103 sensorOffsets.setType = _env->GetMethodID(sensorClass, "setType", "(I)Z");
Peng Xub85b1fe2016-04-20 21:28:33 -0700104 sensorOffsets.setUuid = _env->GetMethodID(sensorClass, "setUuid", "(JJ)V");
Peng Xua35b5532016-01-20 00:05:45 -0800105 sensorOffsets.init = _env->GetMethodID(sensorClass, "<init>", "()V");
106
107 // java.util.List;
108 ListOffsets& listOffsets = gListOffsets;
109 jclass listClass = (jclass) _env->NewGlobalRef(_env->FindClass("java/util/List"));
110 listOffsets.clazz = listClass;
111 listOffsets.add = _env->GetMethodID(listClass, "add", "(Ljava/lang/Object;)Z");
destradaa9ba7c1c2015-02-10 15:04:43 -0800112}
113
114/**
115 * A key comparator predicate.
116 * It is used to intern strings associated with Sensor data.
117 * It defines a 'Strict weak ordering' for the interned strings.
118 */
119class InternedStringCompare {
120public:
121 bool operator()(const String8* string1, const String8* string2) const {
122 if (string1 == NULL) {
123 return string2 != NULL;
124 }
Aravind Akella88445992015-02-26 17:05:28 -0800125 if (string2 == NULL) {
126 return false;
127 }
destradaa9ba7c1c2015-02-10 15:04:43 -0800128 return string1->compare(*string2) < 0;
129 }
130};
131
132/**
133 * A localized interning mechanism for Sensor strings.
134 * We implement our own interning to avoid the overhead of using java.lang.String#intern().
135 * It is common that Vendor, StringType, and RequirePermission data is common between many of the
136 * Sensors, by interning the memory usage to represent Sensors is optimized.
137 */
138static jstring
139getInternedString(JNIEnv *env, const String8* string) {
140 static std::map<const String8*, jstring, InternedStringCompare> internedStrings;
141
142 jstring internedString;
143 std::map<const String8*, jstring>::iterator iterator = internedStrings.find(string);
144 if (iterator != internedStrings.end()) {
145 internedString = iterator->second;
146 } else {
147 jstring localString = env->NewStringUTF(string->string());
148 // we are implementing our own interning so expect these strings to be backed by global refs
149 internedString = (jstring) env->NewGlobalRef(localString);
150 internedStrings.insert(std::make_pair(string, internedString));
151 env->DeleteLocalRef(localString);
152 }
destradaa9ba7c1c2015-02-10 15:04:43 -0800153 return internedString;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800154}
155
Svet Ganovb9d71a62015-04-30 10:38:13 -0700156static jlong
157nativeCreate
158(JNIEnv *env, jclass clazz, jstring opPackageName)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800159{
Svet Ganovb9d71a62015-04-30 10:38:13 -0700160 ScopedUtfChars opPackageNameUtf(env, opPackageName);
Aravind Akella2af91d02015-07-29 18:39:52 -0700161 return (jlong) &SensorManager::getInstanceForPackage(String16(opPackageNameUtf.c_str()));
Svet Ganovb9d71a62015-04-30 10:38:13 -0700162}
163
Peng Xua35b5532016-01-20 00:05:45 -0800164static jobject
165translateNativeSensorToJavaSensor(JNIEnv *env, jobject sensor, const Sensor& nativeSensor) {
166 const SensorOffsets& sensorOffsets(gSensorOffsets);
167
168 if (sensor == NULL) {
169 // Sensor sensor = new Sensor();
170 sensor = env->NewObject(sensorOffsets.clazz, sensorOffsets.init, "");
171 }
172
173 if (sensor != NULL) {
174 jstring name = env->NewStringUTF(nativeSensor.getName().string());
175 jstring vendor = env->NewStringUTF(nativeSensor.getVendor().string());
176 jstring requiredPermission =
177 env->NewStringUTF(nativeSensor.getRequiredPermission().string());
178
179 env->SetObjectField(sensor, sensorOffsets.name, name);
180 env->SetObjectField(sensor, sensorOffsets.vendor, vendor);
181 env->SetIntField(sensor, sensorOffsets.version, nativeSensor.getVersion());
182 env->SetIntField(sensor, sensorOffsets.handle, nativeSensor.getHandle());
183 env->SetFloatField(sensor, sensorOffsets.range, nativeSensor.getMaxValue());
184 env->SetFloatField(sensor, sensorOffsets.resolution, nativeSensor.getResolution());
185 env->SetFloatField(sensor, sensorOffsets.power, nativeSensor.getPowerUsage());
186 env->SetIntField(sensor, sensorOffsets.minDelay, nativeSensor.getMinDelay());
187 env->SetIntField(sensor, sensorOffsets.fifoReservedEventCount,
188 nativeSensor.getFifoReservedEventCount());
189 env->SetIntField(sensor, sensorOffsets.fifoMaxEventCount,
190 nativeSensor.getFifoMaxEventCount());
191 env->SetObjectField(sensor, sensorOffsets.requiredPermission,
192 requiredPermission);
193 env->SetIntField(sensor, sensorOffsets.maxDelay, nativeSensor.getMaxDelay());
194 env->SetIntField(sensor, sensorOffsets.flags, nativeSensor.getFlags());
Peng Xub85b1fe2016-04-20 21:28:33 -0700195
Peng Xua35b5532016-01-20 00:05:45 -0800196 if (env->CallBooleanMethod(sensor, sensorOffsets.setType, nativeSensor.getType())
197 == JNI_FALSE) {
198 jstring stringType = getInternedString(env, &nativeSensor.getStringType());
199 env->SetObjectField(sensor, sensorOffsets.stringType, stringType);
200 }
Peng Xub85b1fe2016-04-20 21:28:33 -0700201
Peng Xu716866d2016-05-25 21:10:00 -0700202 // TODO(b/29547335): Rename "setUuid" method to "setId".
203 int64_t id = nativeSensor.getId();
204 env->CallVoidMethod(sensor, sensorOffsets.setUuid, id, 0);
Peng Xua35b5532016-01-20 00:05:45 -0800205 }
206 return sensor;
207}
208
Aravind Akella516e40e2015-07-01 16:54:24 -0700209static jboolean
210nativeGetSensorAtIndex(JNIEnv *env, jclass clazz, jlong sensorManager, jobject sensor, jint index)
Svet Ganovb9d71a62015-04-30 10:38:13 -0700211{
212 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800213
Mathias Agopian1bf79782010-07-14 23:41:37 -0700214 Sensor const* const* sensorList;
Peng Xua08c9f52016-02-04 22:27:20 -0800215 ssize_t count = mgr->getSensorList(&sensorList);
216 if (ssize_t(index) >= count) {
Aravind Akella516e40e2015-07-01 16:54:24 -0700217 return false;
destradaa9ba7c1c2015-02-10 15:04:43 -0800218 }
Aravind Akellab4c76b12013-06-27 12:04:16 -0700219
Peng Xub85b1fe2016-04-20 21:28:33 -0700220 return translateNativeSensorToJavaSensor(env, sensor, *sensorList[index]) != NULL;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800221}
222
Peng Xua35b5532016-01-20 00:05:45 -0800223static void
224nativeGetDynamicSensors(JNIEnv *env, jclass clazz, jlong sensorManager, jobject sensorList) {
225
226 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
227 const ListOffsets& listOffsets(gListOffsets);
228
229 Vector<Sensor> nativeList;
230
231 mgr->getDynamicSensorList(nativeList);
232
Ben Wagner85ea32b2016-03-16 17:10:42 -0400233 ALOGI("DYNS native SensorManager.getDynamicSensorList return %zu sensors", nativeList.size());
Peng Xua35b5532016-01-20 00:05:45 -0800234 for (size_t i = 0; i < nativeList.size(); ++i) {
235 jobject sensor = translateNativeSensorToJavaSensor(env, NULL, nativeList[i]);
236 // add to list
237 env->CallBooleanMethod(sensorList, listOffsets.add, sensor);
238 }
239}
240
Aravind Akellad123b512015-06-29 12:35:51 -0700241static jboolean nativeIsDataInjectionEnabled(JNIEnv *_env, jclass _this, jlong sensorManager) {
Svet Ganovb9d71a62015-04-30 10:38:13 -0700242 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
Aravind Akellad123b512015-06-29 12:35:51 -0700243 return mgr->isDataInjectionEnabled();
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700244}
245
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800246//----------------------------------------------------------------------------
Mathias Agopiandb772d82013-01-31 19:31:12 -0800247
248class Receiver : public LooperCallback {
249 sp<SensorEventQueue> mSensorQueue;
250 sp<MessageQueue> mMessageQueue;
Jeff Brown3b4049e2015-04-17 15:22:27 -0700251 jobject mReceiverWeakGlobal;
Peng Xua2958352016-01-26 18:42:17 -0800252 jfloatArray mFloatScratch;
253 jintArray mIntScratch;
Mathias Agopiandb772d82013-01-31 19:31:12 -0800254public:
255 Receiver(const sp<SensorEventQueue>& sensorQueue,
256 const sp<MessageQueue>& messageQueue,
Peng Xua2958352016-01-26 18:42:17 -0800257 jobject receiverWeak) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800258 JNIEnv* env = AndroidRuntime::getJNIEnv();
259 mSensorQueue = sensorQueue;
260 mMessageQueue = messageQueue;
Jeff Brown3b4049e2015-04-17 15:22:27 -0700261 mReceiverWeakGlobal = env->NewGlobalRef(receiverWeak);
Peng Xua2958352016-01-26 18:42:17 -0800262
263 mIntScratch = (jintArray) env->NewGlobalRef(env->NewIntArray(16));
264 mFloatScratch = (jfloatArray) env->NewGlobalRef(env->NewFloatArray(16));
Mathias Agopiandb772d82013-01-31 19:31:12 -0800265 }
266 ~Receiver() {
267 JNIEnv* env = AndroidRuntime::getJNIEnv();
Jeff Brown3b4049e2015-04-17 15:22:27 -0700268 env->DeleteGlobalRef(mReceiverWeakGlobal);
Peng Xua2958352016-01-26 18:42:17 -0800269 env->DeleteGlobalRef(mFloatScratch);
270 env->DeleteGlobalRef(mIntScratch);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800271 }
272 sp<SensorEventQueue> getSensorEventQueue() const {
273 return mSensorQueue;
274 }
275
276 void destroy() {
277 mMessageQueue->getLooper()->removeFd( mSensorQueue->getFd() );
278 }
279
280private:
281 virtual void onFirstRef() {
282 LooperCallback::onFirstRef();
283 mMessageQueue->getLooper()->addFd(mSensorQueue->getFd(), 0,
284 ALOOPER_EVENT_INPUT, this, mSensorQueue.get());
285 }
286
287 virtual int handleEvent(int fd, int events, void* data) {
288 JNIEnv* env = AndroidRuntime::getJNIEnv();
289 sp<SensorEventQueue> q = reinterpret_cast<SensorEventQueue *>(data);
Jeff Brown3b4049e2015-04-17 15:22:27 -0700290 ScopedLocalRef<jobject> receiverObj(env, jniGetReferent(env, mReceiverWeakGlobal));
291
Mathias Agopiandb772d82013-01-31 19:31:12 -0800292 ssize_t n;
293 ASensorEvent buffer[16];
294 while ((n = q->read(buffer, 16)) > 0) {
295 for (int i=0 ; i<n ; i++) {
Aravind Akella0288ca62014-07-28 14:53:44 -0700296 if (buffer[i].type == SENSOR_TYPE_STEP_COUNTER) {
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700297 // step-counter returns a uint64, but the java API only deals with floats
298 float value = float(buffer[i].u64.step_counter);
Peng Xua2958352016-01-26 18:42:17 -0800299 env->SetFloatArrayRegion(mFloatScratch, 0, 1, &value);
Peng Xua35b5532016-01-20 00:05:45 -0800300 } else if (buffer[i].type == SENSOR_TYPE_DYNAMIC_SENSOR_META) {
301 float value[2];
302 value[0] = buffer[i].dynamic_sensor_meta.connected ? 1.f: 0.f;
303 value[1] = float(buffer[i].dynamic_sensor_meta.handle);
Peng Xua2958352016-01-26 18:42:17 -0800304 env->SetFloatArrayRegion(mFloatScratch, 0, 2, value);
305 } else if (buffer[i].type == SENSOR_TYPE_ADDITIONAL_INFO) {
306 env->SetIntArrayRegion(mIntScratch, 0, 14,
307 buffer[i].additional_info.data_int32);
308 env->SetFloatArrayRegion(mFloatScratch, 0, 14,
309 buffer[i].additional_info.data_float);
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700310 } else {
Peng Xua2958352016-01-26 18:42:17 -0800311 env->SetFloatArrayRegion(mFloatScratch, 0, 16, buffer[i].data);
Jean-Baptiste Queru3c9c0912013-08-05 11:03:28 -0700312 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800313
Aravind Akellab4c76b12013-06-27 12:04:16 -0700314 if (buffer[i].type == SENSOR_TYPE_META_DATA) {
315 // This is a flush complete sensor event. Call dispatchFlushCompleteEvent
316 // method.
Jeff Brown3b4049e2015-04-17 15:22:27 -0700317 if (receiverObj.get()) {
318 env->CallVoidMethod(receiverObj.get(),
319 gBaseEventQueueClassInfo.dispatchFlushCompleteEvent,
320 buffer[i].meta_data.sensor);
321 }
Peng Xua2958352016-01-26 18:42:17 -0800322 } else if (buffer[i].type == SENSOR_TYPE_ADDITIONAL_INFO) {
323 // This is a flush complete sensor event. Call dispatchAdditionalInfoEvent
324 // method.
325 if (receiverObj.get()) {
326 int type = buffer[i].additional_info.type;
327 int serial = buffer[i].additional_info.serial;
328 env->CallVoidMethod(receiverObj.get(),
329 gBaseEventQueueClassInfo.dispatchAdditionalInfoEvent,
330 buffer[i].sensor,
331 type, serial,
332 mFloatScratch,
333 mIntScratch,
334 buffer[i].timestamp);
335 }
336 }else {
Etienne Le Grandaf805102014-05-16 12:21:41 -0700337 int8_t status;
338 switch (buffer[i].type) {
339 case SENSOR_TYPE_ORIENTATION:
340 case SENSOR_TYPE_MAGNETIC_FIELD:
341 case SENSOR_TYPE_ACCELEROMETER:
342 case SENSOR_TYPE_GYROSCOPE:
Aravind Akellaf38226a2015-06-23 10:32:16 -0700343 case SENSOR_TYPE_GRAVITY:
344 case SENSOR_TYPE_LINEAR_ACCELERATION:
Etienne Le Grandaf805102014-05-16 12:21:41 -0700345 status = buffer[i].vector.status;
346 break;
347 case SENSOR_TYPE_HEART_RATE:
348 status = buffer[i].heart_rate.status;
349 break;
350 default:
351 status = SENSOR_STATUS_ACCURACY_HIGH;
352 break;
353 }
Jeff Brown3b4049e2015-04-17 15:22:27 -0700354 if (receiverObj.get()) {
355 env->CallVoidMethod(receiverObj.get(),
356 gBaseEventQueueClassInfo.dispatchSensorEvent,
357 buffer[i].sensor,
Peng Xua2958352016-01-26 18:42:17 -0800358 mFloatScratch,
Jeff Brown3b4049e2015-04-17 15:22:27 -0700359 status,
360 buffer[i].timestamp);
361 }
Aravind Akellab4c76b12013-06-27 12:04:16 -0700362 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800363 if (env->ExceptionCheck()) {
Aravind Akella35187bd2014-02-11 18:44:42 -0800364 mSensorQueue->sendAck(buffer, n);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800365 ALOGE("Exception dispatching input event.");
366 return 1;
367 }
368 }
Aravind Akella35187bd2014-02-11 18:44:42 -0800369 mSensorQueue->sendAck(buffer, n);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800370 }
371 if (n<0 && n != -EAGAIN) {
372 // FIXME: error receiving events, what to do in this case?
373 }
Mathias Agopiandb772d82013-01-31 19:31:12 -0800374 return 1;
375 }
376};
377
Svet Ganovb9d71a62015-04-30 10:38:13 -0700378static jlong nativeInitSensorEventQueue(JNIEnv *env, jclass clazz, jlong sensorManager,
Peng Xua2958352016-01-26 18:42:17 -0800379 jobject eventQWeak, jobject msgQ, jstring packageName, jint mode) {
Svet Ganovb9d71a62015-04-30 10:38:13 -0700380 SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager);
Aravind Akella88445992015-02-26 17:05:28 -0800381 ScopedUtfChars packageUtf(env, packageName);
382 String8 clientName(packageUtf.c_str());
Svet Ganovb9d71a62015-04-30 10:38:13 -0700383 sp<SensorEventQueue> queue(mgr->createEventQueue(clientName, mode));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800384
Mathias Agopiandb772d82013-01-31 19:31:12 -0800385 sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, msgQ);
386 if (messageQueue == NULL) {
387 jniThrowRuntimeException(env, "MessageQueue is not initialized.");
388 return 0;
Mathias Agopian0b6d77b2012-05-07 18:27:06 -0700389 }
Mathias Agopian1bf79782010-07-14 23:41:37 -0700390
Peng Xua2958352016-01-26 18:42:17 -0800391 sp<Receiver> receiver = new Receiver(queue, messageQueue, eventQWeak);
Mathias Agopianb1d90c82013-03-06 17:45:42 -0800392 receiver->incStrong((void*)nativeInitSensorEventQueue);
Ashok Bhat4838e332014-01-03 14:37:19 +0000393 return jlong(receiver.get());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800394}
395
Ashok Bhat4838e332014-01-03 14:37:19 +0000396static jint nativeEnableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle, jint rate_us,
Aravind Akella88445992015-02-26 17:05:28 -0800397 jint maxBatchReportLatency) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800398 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
Aravind Akellab4c76b12013-06-27 12:04:16 -0700399 return receiver->getSensorEventQueue()->enableSensor(handle, rate_us, maxBatchReportLatency,
Aravind Akella88445992015-02-26 17:05:28 -0800400 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800401}
402
Ashok Bhat4838e332014-01-03 14:37:19 +0000403static jint nativeDisableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800404 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
405 return receiver->getSensorEventQueue()->disableSensor(handle);
406}
Mathias Agopian1bf79782010-07-14 23:41:37 -0700407
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700408static void nativeDestroySensorEventQueue(JNIEnv *env, jclass clazz, jlong eventQ) {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800409 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
410 receiver->destroy();
Mathias Agopianb1d90c82013-03-06 17:45:42 -0800411 receiver->decStrong((void*)nativeInitSensorEventQueue);
Mathias Agopiandb772d82013-01-31 19:31:12 -0800412}
Mathias Agopian1bf79782010-07-14 23:41:37 -0700413
Ashok Bhat4838e332014-01-03 14:37:19 +0000414static jint nativeFlushSensor(JNIEnv *env, jclass clazz, jlong eventQ) {
Aravind Akellab4c76b12013-06-27 12:04:16 -0700415 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
Aravind Akella4bdc37d2013-10-01 17:58:35 -0700416 return receiver->getSensorEventQueue()->flush();
Aravind Akellab4c76b12013-06-27 12:04:16 -0700417}
Mathias Agopiandb772d82013-01-31 19:31:12 -0800418
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700419static jint nativeInjectSensorData(JNIEnv *env, jclass clazz, jlong eventQ, jint handle,
420 jfloatArray values, jint accuracy, jlong timestamp) {
421 sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ));
422 // Create a sensor_event from the above data which can be injected into the HAL.
423 ASensorEvent sensor_event;
424 memset(&sensor_event, 0, sizeof(sensor_event));
425 sensor_event.sensor = handle;
426 sensor_event.timestamp = timestamp;
427 env->GetFloatArrayRegion(values, 0, env->GetArrayLength(values), sensor_event.data);
428 return receiver->getSensorEventQueue()->injectSensorEvent(sensor_event);
429}
Mathias Agopiandb772d82013-01-31 19:31:12 -0800430//----------------------------------------------------------------------------
431
Daniel Micay76f6a862015-09-19 17:31:01 -0400432static const JNINativeMethod gSystemSensorManagerMethods[] = {
Mathias Agopiandb772d82013-01-31 19:31:12 -0800433 {"nativeClassInit",
434 "()V",
435 (void*)nativeClassInit },
Svet Ganovb9d71a62015-04-30 10:38:13 -0700436 {"nativeCreate",
437 "(Ljava/lang/String;)J",
438 (void*)nativeCreate },
Mathias Agopiandb772d82013-01-31 19:31:12 -0800439
Aravind Akella516e40e2015-07-01 16:54:24 -0700440 {"nativeGetSensorAtIndex",
441 "(JLandroid/hardware/Sensor;I)Z",
442 (void*)nativeGetSensorAtIndex },
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700443
Peng Xua35b5532016-01-20 00:05:45 -0800444 {"nativeGetDynamicSensors",
445 "(JLjava/util/List;)V",
446 (void*)nativeGetDynamicSensors },
447
Aravind Akellad123b512015-06-29 12:35:51 -0700448 {"nativeIsDataInjectionEnabled",
449 "(J)Z",
450 (void*)nativeIsDataInjectionEnabled},
Mathias Agopiandb772d82013-01-31 19:31:12 -0800451};
452
Daniel Micay76f6a862015-09-19 17:31:01 -0400453static const JNINativeMethod gBaseEventQueueMethods[] = {
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -0800454 {"nativeInitBaseEventQueue",
Peng Xua2958352016-01-26 18:42:17 -0800455 "(JLjava/lang/ref/WeakReference;Landroid/os/MessageQueue;Ljava/lang/String;ILjava/lang/String;)J",
Svet Ganovb9d71a62015-04-30 10:38:13 -0700456 (void*)nativeInitSensorEventQueue },
Mathias Agopiandb772d82013-01-31 19:31:12 -0800457
458 {"nativeEnableSensor",
Aravind Akella88445992015-02-26 17:05:28 -0800459 "(JIII)I",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800460 (void*)nativeEnableSensor },
461
462 {"nativeDisableSensor",
Ashok Bhat4838e332014-01-03 14:37:19 +0000463 "(JI)I",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800464 (void*)nativeDisableSensor },
465
466 {"nativeDestroySensorEventQueue",
Ashok Bhat4838e332014-01-03 14:37:19 +0000467 "(J)V",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800468 (void*)nativeDestroySensorEventQueue },
Aravind Akellab4c76b12013-06-27 12:04:16 -0700469
470 {"nativeFlushSensor",
Ashok Bhat4838e332014-01-03 14:37:19 +0000471 "(J)I",
Aravind Akellab4c76b12013-06-27 12:04:16 -0700472 (void*)nativeFlushSensor },
Aravind Akella18ebf732b2015-04-20 17:39:51 -0700473
474 {"nativeInjectSensorData",
475 "(JI[FIJ)I",
476 (void*)nativeInjectSensorData },
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800477};
478
Peng Xu4a4baef2016-03-16 16:56:39 -0700479} //unnamed namespace
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800480
481int register_android_hardware_SensorManager(JNIEnv *env)
482{
Andreas Gampe987f79f2014-11-18 17:29:46 -0800483 RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager",
Mathias Agopiandb772d82013-01-31 19:31:12 -0800484 gSystemSensorManagerMethods, NELEM(gSystemSensorManagerMethods));
485
Andreas Gampe987f79f2014-11-18 17:29:46 -0800486 RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager$BaseEventQueue",
Jaikumar Ganesh9a8df4d2013-02-12 16:31:32 -0800487 gBaseEventQueueMethods, NELEM(gBaseEventQueueMethods));
Mathias Agopiandb772d82013-01-31 19:31:12 -0800488
Andreas Gampe987f79f2014-11-18 17:29:46 -0800489 gBaseEventQueueClassInfo.clazz = FindClassOrDie(env,
490 "android/hardware/SystemSensorManager$BaseEventQueue");
Mathias Agopiandb772d82013-01-31 19:31:12 -0800491
Andreas Gampe987f79f2014-11-18 17:29:46 -0800492 gBaseEventQueueClassInfo.dispatchSensorEvent = GetMethodIDOrDie(env,
493 gBaseEventQueueClassInfo.clazz, "dispatchSensorEvent", "(I[FIJ)V");
Mathias Agopiandb772d82013-01-31 19:31:12 -0800494
Andreas Gampe987f79f2014-11-18 17:29:46 -0800495 gBaseEventQueueClassInfo.dispatchFlushCompleteEvent = GetMethodIDOrDie(env,
496 gBaseEventQueueClassInfo.clazz, "dispatchFlushCompleteEvent", "(I)V");
Aravind Akellab4c76b12013-06-27 12:04:16 -0700497
Peng Xua2958352016-01-26 18:42:17 -0800498 gBaseEventQueueClassInfo.dispatchAdditionalInfoEvent = GetMethodIDOrDie(env,
499 gBaseEventQueueClassInfo.clazz, "dispatchAdditionalInfoEvent", "(III[F[I)V");
500
Mathias Agopiandb772d82013-01-31 19:31:12 -0800501 return 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800502}