The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 1 | /* |
| 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 Lockwood | 270e87f | 2009-05-22 10:02:14 -0400 | [diff] [blame] | 16 | #define LOG_TAG "SensorManager" |
| 17 | |
Peng Xu | d5ec5f9 | 2016-04-27 14:49:25 -0700 | [diff] [blame] | 18 | #include "JNIHelp.h" |
| 19 | #include "android_os_MessageQueue.h" |
| 20 | #include "core_jni_helpers.h" |
| 21 | #include "jni.h" |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 22 | |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 23 | #include <ScopedUtfChars.h> |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 24 | #include <ScopedLocalRef.h> |
Peng Xu | d5ec5f9 | 2016-04-27 14:49:25 -0700 | [diff] [blame] | 25 | #include <android_runtime/AndroidRuntime.h> |
| 26 | #include <gui/Sensor.h> |
| 27 | #include <gui/SensorEventQueue.h> |
| 28 | #include <gui/SensorManager.h> |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 29 | #include <utils/Log.h> |
| 30 | #include <utils/Looper.h> |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 31 | #include <utils/Vector.h> |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 32 | |
Peng Xu | d5ec5f9 | 2016-04-27 14:49:25 -0700 | [diff] [blame] | 33 | #include <map> |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 34 | |
Peng Xu | 4a4baef | 2016-03-16 16:56:39 -0700 | [diff] [blame] | 35 | namespace { |
| 36 | |
| 37 | using namespace android; |
| 38 | |
| 39 | struct { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 40 | jclass clazz; |
| 41 | jmethodID dispatchSensorEvent; |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 42 | jmethodID dispatchFlushCompleteEvent; |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 43 | jmethodID dispatchAdditionalInfoEvent; |
Jaikumar Ganesh | 9a8df4d | 2013-02-12 16:31:32 -0800 | [diff] [blame] | 44 | } gBaseEventQueueClassInfo; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 45 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 46 | struct SensorOffsets |
| 47 | { |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 48 | jclass clazz; |
Peng Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 49 | //fields |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 50 | jfieldID name; |
| 51 | jfieldID vendor; |
| 52 | jfieldID version; |
| 53 | jfieldID handle; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 54 | jfieldID range; |
| 55 | jfieldID resolution; |
| 56 | jfieldID power; |
Mathias Agopian | 050b562 | 2010-07-29 16:51:38 -0700 | [diff] [blame] | 57 | jfieldID minDelay; |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 58 | jfieldID fifoReservedEventCount; |
| 59 | jfieldID fifoMaxEventCount; |
Aravind Akella | cd9a7bb | 2014-04-07 22:55:21 +0000 | [diff] [blame] | 60 | jfieldID stringType; |
| 61 | jfieldID requiredPermission; |
Aravind Akella | 2f8b914 | 2014-05-12 14:39:29 -0700 | [diff] [blame] | 62 | jfieldID maxDelay; |
Aravind Akella | 2790035 | 2014-06-03 19:20:42 -0700 | [diff] [blame] | 63 | jfieldID flags; |
Peng Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 64 | //methods |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 65 | jmethodID setType; |
Peng Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 66 | jmethodID setUuid; |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 67 | jmethodID init; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 68 | } gSensorOffsets; |
| 69 | |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 70 | struct ListOffsets { |
| 71 | jclass clazz; |
| 72 | jmethodID add; |
| 73 | } gListOffsets; |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 74 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 75 | /* |
Peng Xu | 4a4baef | 2016-03-16 16:56:39 -0700 | [diff] [blame] | 76 | * nativeClassInit is not inteneded to be thread-safe. It should be called before other native... |
| 77 | * functions (except nativeCreate). |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 78 | */ |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 79 | static void |
| 80 | nativeClassInit (JNIEnv *_env, jclass _this) |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 81 | { |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 82 | //android.hardware.Sensor |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 83 | SensorOffsets& sensorOffsets = gSensorOffsets; |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 84 | jclass sensorClass = (jclass) _env->NewGlobalRef(_env->FindClass("android/hardware/Sensor")); |
| 85 | sensorOffsets.clazz = sensorClass; |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 86 | 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 Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 90 | 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 Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 94 | sensorOffsets.fifoReservedEventCount = |
| 95 | _env->GetFieldID(sensorClass, "mFifoReservedEventCount", "I"); |
| 96 | sensorOffsets.fifoMaxEventCount = _env->GetFieldID(sensorClass, "mFifoMaxEventCount", "I"); |
Aravind Akella | cd9a7bb | 2014-04-07 22:55:21 +0000 | [diff] [blame] | 97 | sensorOffsets.stringType = _env->GetFieldID(sensorClass, "mStringType", "Ljava/lang/String;"); |
| 98 | sensorOffsets.requiredPermission = _env->GetFieldID(sensorClass, "mRequiredPermission", |
| 99 | "Ljava/lang/String;"); |
Aravind Akella | 2f8b914 | 2014-05-12 14:39:29 -0700 | [diff] [blame] | 100 | sensorOffsets.maxDelay = _env->GetFieldID(sensorClass, "mMaxDelay", "I"); |
Aravind Akella | 2790035 | 2014-06-03 19:20:42 -0700 | [diff] [blame] | 101 | sensorOffsets.flags = _env->GetFieldID(sensorClass, "mFlags", "I"); |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 102 | |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 103 | sensorOffsets.setType = _env->GetMethodID(sensorClass, "setType", "(I)Z"); |
Peng Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 104 | sensorOffsets.setUuid = _env->GetMethodID(sensorClass, "setUuid", "(JJ)V"); |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 105 | 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"); |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 112 | } |
| 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 | */ |
| 119 | class InternedStringCompare { |
| 120 | public: |
| 121 | bool operator()(const String8* string1, const String8* string2) const { |
| 122 | if (string1 == NULL) { |
| 123 | return string2 != NULL; |
| 124 | } |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 125 | if (string2 == NULL) { |
| 126 | return false; |
| 127 | } |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 128 | 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 | */ |
| 138 | static jstring |
| 139 | getInternedString(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 | } |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 153 | return internedString; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 154 | } |
| 155 | |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 156 | static jlong |
| 157 | nativeCreate |
| 158 | (JNIEnv *env, jclass clazz, jstring opPackageName) |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 159 | { |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 160 | ScopedUtfChars opPackageNameUtf(env, opPackageName); |
Aravind Akella | 2af91d0 | 2015-07-29 18:39:52 -0700 | [diff] [blame] | 161 | return (jlong) &SensorManager::getInstanceForPackage(String16(opPackageNameUtf.c_str())); |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 162 | } |
| 163 | |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 164 | static jobject |
| 165 | translateNativeSensorToJavaSensor(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 Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 195 | |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 196 | 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 Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 201 | |
Peng Xu | 716866d | 2016-05-25 21:10:00 -0700 | [diff] [blame] | 202 | // TODO(b/29547335): Rename "setUuid" method to "setId". |
| 203 | int64_t id = nativeSensor.getId(); |
| 204 | env->CallVoidMethod(sensor, sensorOffsets.setUuid, id, 0); |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 205 | } |
| 206 | return sensor; |
| 207 | } |
| 208 | |
Aravind Akella | 516e40e | 2015-07-01 16:54:24 -0700 | [diff] [blame] | 209 | static jboolean |
| 210 | nativeGetSensorAtIndex(JNIEnv *env, jclass clazz, jlong sensorManager, jobject sensor, jint index) |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 211 | { |
| 212 | SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 213 | |
Mathias Agopian | 1bf7978 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 214 | Sensor const* const* sensorList; |
Peng Xu | a08c9f5 | 2016-02-04 22:27:20 -0800 | [diff] [blame] | 215 | ssize_t count = mgr->getSensorList(&sensorList); |
| 216 | if (ssize_t(index) >= count) { |
Aravind Akella | 516e40e | 2015-07-01 16:54:24 -0700 | [diff] [blame] | 217 | return false; |
destradaa | 9ba7c1c | 2015-02-10 15:04:43 -0800 | [diff] [blame] | 218 | } |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 219 | |
Peng Xu | b85b1fe | 2016-04-20 21:28:33 -0700 | [diff] [blame] | 220 | return translateNativeSensorToJavaSensor(env, sensor, *sensorList[index]) != NULL; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 221 | } |
| 222 | |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 223 | static void |
| 224 | nativeGetDynamicSensors(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 Wagner | 85ea32b | 2016-03-16 17:10:42 -0400 | [diff] [blame] | 233 | ALOGI("DYNS native SensorManager.getDynamicSensorList return %zu sensors", nativeList.size()); |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 234 | 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 Akella | d123b51 | 2015-06-29 12:35:51 -0700 | [diff] [blame] | 241 | static jboolean nativeIsDataInjectionEnabled(JNIEnv *_env, jclass _this, jlong sensorManager) { |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 242 | SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager); |
Aravind Akella | d123b51 | 2015-06-29 12:35:51 -0700 | [diff] [blame] | 243 | return mgr->isDataInjectionEnabled(); |
Aravind Akella | 18ebf732b | 2015-04-20 17:39:51 -0700 | [diff] [blame] | 244 | } |
| 245 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 246 | //---------------------------------------------------------------------------- |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 247 | |
| 248 | class Receiver : public LooperCallback { |
| 249 | sp<SensorEventQueue> mSensorQueue; |
| 250 | sp<MessageQueue> mMessageQueue; |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 251 | jobject mReceiverWeakGlobal; |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 252 | jfloatArray mFloatScratch; |
| 253 | jintArray mIntScratch; |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 254 | public: |
| 255 | Receiver(const sp<SensorEventQueue>& sensorQueue, |
| 256 | const sp<MessageQueue>& messageQueue, |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 257 | jobject receiverWeak) { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 258 | JNIEnv* env = AndroidRuntime::getJNIEnv(); |
| 259 | mSensorQueue = sensorQueue; |
| 260 | mMessageQueue = messageQueue; |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 261 | mReceiverWeakGlobal = env->NewGlobalRef(receiverWeak); |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 262 | |
| 263 | mIntScratch = (jintArray) env->NewGlobalRef(env->NewIntArray(16)); |
| 264 | mFloatScratch = (jfloatArray) env->NewGlobalRef(env->NewFloatArray(16)); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 265 | } |
| 266 | ~Receiver() { |
| 267 | JNIEnv* env = AndroidRuntime::getJNIEnv(); |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 268 | env->DeleteGlobalRef(mReceiverWeakGlobal); |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 269 | env->DeleteGlobalRef(mFloatScratch); |
| 270 | env->DeleteGlobalRef(mIntScratch); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 271 | } |
| 272 | sp<SensorEventQueue> getSensorEventQueue() const { |
| 273 | return mSensorQueue; |
| 274 | } |
| 275 | |
| 276 | void destroy() { |
| 277 | mMessageQueue->getLooper()->removeFd( mSensorQueue->getFd() ); |
| 278 | } |
| 279 | |
| 280 | private: |
| 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 Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 290 | ScopedLocalRef<jobject> receiverObj(env, jniGetReferent(env, mReceiverWeakGlobal)); |
| 291 | |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 292 | ssize_t n; |
| 293 | ASensorEvent buffer[16]; |
| 294 | while ((n = q->read(buffer, 16)) > 0) { |
| 295 | for (int i=0 ; i<n ; i++) { |
Aravind Akella | 0288ca6 | 2014-07-28 14:53:44 -0700 | [diff] [blame] | 296 | if (buffer[i].type == SENSOR_TYPE_STEP_COUNTER) { |
Jean-Baptiste Queru | 3c9c091 | 2013-08-05 11:03:28 -0700 | [diff] [blame] | 297 | // step-counter returns a uint64, but the java API only deals with floats |
| 298 | float value = float(buffer[i].u64.step_counter); |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 299 | env->SetFloatArrayRegion(mFloatScratch, 0, 1, &value); |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 300 | } 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 Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 304 | 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 Queru | 3c9c091 | 2013-08-05 11:03:28 -0700 | [diff] [blame] | 310 | } else { |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 311 | env->SetFloatArrayRegion(mFloatScratch, 0, 16, buffer[i].data); |
Jean-Baptiste Queru | 3c9c091 | 2013-08-05 11:03:28 -0700 | [diff] [blame] | 312 | } |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 313 | |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 314 | if (buffer[i].type == SENSOR_TYPE_META_DATA) { |
| 315 | // This is a flush complete sensor event. Call dispatchFlushCompleteEvent |
| 316 | // method. |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 317 | if (receiverObj.get()) { |
| 318 | env->CallVoidMethod(receiverObj.get(), |
| 319 | gBaseEventQueueClassInfo.dispatchFlushCompleteEvent, |
| 320 | buffer[i].meta_data.sensor); |
| 321 | } |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 322 | } 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 Grand | af80510 | 2014-05-16 12:21:41 -0700 | [diff] [blame] | 337 | 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 Akella | f38226a | 2015-06-23 10:32:16 -0700 | [diff] [blame] | 343 | case SENSOR_TYPE_GRAVITY: |
| 344 | case SENSOR_TYPE_LINEAR_ACCELERATION: |
Etienne Le Grand | af80510 | 2014-05-16 12:21:41 -0700 | [diff] [blame] | 345 | 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 Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 354 | if (receiverObj.get()) { |
| 355 | env->CallVoidMethod(receiverObj.get(), |
| 356 | gBaseEventQueueClassInfo.dispatchSensorEvent, |
| 357 | buffer[i].sensor, |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 358 | mFloatScratch, |
Jeff Brown | 3b4049e | 2015-04-17 15:22:27 -0700 | [diff] [blame] | 359 | status, |
| 360 | buffer[i].timestamp); |
| 361 | } |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 362 | } |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 363 | if (env->ExceptionCheck()) { |
Aravind Akella | 35187bd | 2014-02-11 18:44:42 -0800 | [diff] [blame] | 364 | mSensorQueue->sendAck(buffer, n); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 365 | ALOGE("Exception dispatching input event."); |
| 366 | return 1; |
| 367 | } |
| 368 | } |
Aravind Akella | 35187bd | 2014-02-11 18:44:42 -0800 | [diff] [blame] | 369 | mSensorQueue->sendAck(buffer, n); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 370 | } |
| 371 | if (n<0 && n != -EAGAIN) { |
| 372 | // FIXME: error receiving events, what to do in this case? |
| 373 | } |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 374 | return 1; |
| 375 | } |
| 376 | }; |
| 377 | |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 378 | static jlong nativeInitSensorEventQueue(JNIEnv *env, jclass clazz, jlong sensorManager, |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 379 | jobject eventQWeak, jobject msgQ, jstring packageName, jint mode) { |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 380 | SensorManager* mgr = reinterpret_cast<SensorManager*>(sensorManager); |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 381 | ScopedUtfChars packageUtf(env, packageName); |
| 382 | String8 clientName(packageUtf.c_str()); |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 383 | sp<SensorEventQueue> queue(mgr->createEventQueue(clientName, mode)); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 384 | |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 385 | sp<MessageQueue> messageQueue = android_os_MessageQueue_getMessageQueue(env, msgQ); |
| 386 | if (messageQueue == NULL) { |
| 387 | jniThrowRuntimeException(env, "MessageQueue is not initialized."); |
| 388 | return 0; |
Mathias Agopian | 0b6d77b | 2012-05-07 18:27:06 -0700 | [diff] [blame] | 389 | } |
Mathias Agopian | 1bf7978 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 390 | |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 391 | sp<Receiver> receiver = new Receiver(queue, messageQueue, eventQWeak); |
Mathias Agopian | b1d90c8 | 2013-03-06 17:45:42 -0800 | [diff] [blame] | 392 | receiver->incStrong((void*)nativeInitSensorEventQueue); |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 393 | return jlong(receiver.get()); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 394 | } |
| 395 | |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 396 | static jint nativeEnableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle, jint rate_us, |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 397 | jint maxBatchReportLatency) { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 398 | sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ)); |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 399 | return receiver->getSensorEventQueue()->enableSensor(handle, rate_us, maxBatchReportLatency, |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 400 | 0); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 401 | } |
| 402 | |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 403 | static jint nativeDisableSensor(JNIEnv *env, jclass clazz, jlong eventQ, jint handle) { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 404 | sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ)); |
| 405 | return receiver->getSensorEventQueue()->disableSensor(handle); |
| 406 | } |
Mathias Agopian | 1bf7978 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 407 | |
Aravind Akella | 18ebf732b | 2015-04-20 17:39:51 -0700 | [diff] [blame] | 408 | static void nativeDestroySensorEventQueue(JNIEnv *env, jclass clazz, jlong eventQ) { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 409 | sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ)); |
| 410 | receiver->destroy(); |
Mathias Agopian | b1d90c8 | 2013-03-06 17:45:42 -0800 | [diff] [blame] | 411 | receiver->decStrong((void*)nativeInitSensorEventQueue); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 412 | } |
Mathias Agopian | 1bf7978 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 413 | |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 414 | static jint nativeFlushSensor(JNIEnv *env, jclass clazz, jlong eventQ) { |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 415 | sp<Receiver> receiver(reinterpret_cast<Receiver *>(eventQ)); |
Aravind Akella | 4bdc37d | 2013-10-01 17:58:35 -0700 | [diff] [blame] | 416 | return receiver->getSensorEventQueue()->flush(); |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 417 | } |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 418 | |
Aravind Akella | 18ebf732b | 2015-04-20 17:39:51 -0700 | [diff] [blame] | 419 | static 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 Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 430 | //---------------------------------------------------------------------------- |
| 431 | |
Daniel Micay | 76f6a86 | 2015-09-19 17:31:01 -0400 | [diff] [blame] | 432 | static const JNINativeMethod gSystemSensorManagerMethods[] = { |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 433 | {"nativeClassInit", |
| 434 | "()V", |
| 435 | (void*)nativeClassInit }, |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 436 | {"nativeCreate", |
| 437 | "(Ljava/lang/String;)J", |
| 438 | (void*)nativeCreate }, |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 439 | |
Aravind Akella | 516e40e | 2015-07-01 16:54:24 -0700 | [diff] [blame] | 440 | {"nativeGetSensorAtIndex", |
| 441 | "(JLandroid/hardware/Sensor;I)Z", |
| 442 | (void*)nativeGetSensorAtIndex }, |
Aravind Akella | 18ebf732b | 2015-04-20 17:39:51 -0700 | [diff] [blame] | 443 | |
Peng Xu | a35b553 | 2016-01-20 00:05:45 -0800 | [diff] [blame] | 444 | {"nativeGetDynamicSensors", |
| 445 | "(JLjava/util/List;)V", |
| 446 | (void*)nativeGetDynamicSensors }, |
| 447 | |
Aravind Akella | d123b51 | 2015-06-29 12:35:51 -0700 | [diff] [blame] | 448 | {"nativeIsDataInjectionEnabled", |
| 449 | "(J)Z", |
| 450 | (void*)nativeIsDataInjectionEnabled}, |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 451 | }; |
| 452 | |
Daniel Micay | 76f6a86 | 2015-09-19 17:31:01 -0400 | [diff] [blame] | 453 | static const JNINativeMethod gBaseEventQueueMethods[] = { |
Jaikumar Ganesh | 9a8df4d | 2013-02-12 16:31:32 -0800 | [diff] [blame] | 454 | {"nativeInitBaseEventQueue", |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 455 | "(JLjava/lang/ref/WeakReference;Landroid/os/MessageQueue;Ljava/lang/String;ILjava/lang/String;)J", |
Svet Ganov | b9d71a6 | 2015-04-30 10:38:13 -0700 | [diff] [blame] | 456 | (void*)nativeInitSensorEventQueue }, |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 457 | |
| 458 | {"nativeEnableSensor", |
Aravind Akella | 8844599 | 2015-02-26 17:05:28 -0800 | [diff] [blame] | 459 | "(JIII)I", |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 460 | (void*)nativeEnableSensor }, |
| 461 | |
| 462 | {"nativeDisableSensor", |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 463 | "(JI)I", |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 464 | (void*)nativeDisableSensor }, |
| 465 | |
| 466 | {"nativeDestroySensorEventQueue", |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 467 | "(J)V", |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 468 | (void*)nativeDestroySensorEventQueue }, |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 469 | |
| 470 | {"nativeFlushSensor", |
Ashok Bhat | 4838e33 | 2014-01-03 14:37:19 +0000 | [diff] [blame] | 471 | "(J)I", |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 472 | (void*)nativeFlushSensor }, |
Aravind Akella | 18ebf732b | 2015-04-20 17:39:51 -0700 | [diff] [blame] | 473 | |
| 474 | {"nativeInjectSensorData", |
| 475 | "(JI[FIJ)I", |
| 476 | (void*)nativeInjectSensorData }, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 477 | }; |
| 478 | |
Peng Xu | 4a4baef | 2016-03-16 16:56:39 -0700 | [diff] [blame] | 479 | } //unnamed namespace |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 480 | |
| 481 | int register_android_hardware_SensorManager(JNIEnv *env) |
| 482 | { |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 483 | RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager", |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 484 | gSystemSensorManagerMethods, NELEM(gSystemSensorManagerMethods)); |
| 485 | |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 486 | RegisterMethodsOrDie(env, "android/hardware/SystemSensorManager$BaseEventQueue", |
Jaikumar Ganesh | 9a8df4d | 2013-02-12 16:31:32 -0800 | [diff] [blame] | 487 | gBaseEventQueueMethods, NELEM(gBaseEventQueueMethods)); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 488 | |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 489 | gBaseEventQueueClassInfo.clazz = FindClassOrDie(env, |
| 490 | "android/hardware/SystemSensorManager$BaseEventQueue"); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 491 | |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 492 | gBaseEventQueueClassInfo.dispatchSensorEvent = GetMethodIDOrDie(env, |
| 493 | gBaseEventQueueClassInfo.clazz, "dispatchSensorEvent", "(I[FIJ)V"); |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 494 | |
Andreas Gampe | 987f79f | 2014-11-18 17:29:46 -0800 | [diff] [blame] | 495 | gBaseEventQueueClassInfo.dispatchFlushCompleteEvent = GetMethodIDOrDie(env, |
| 496 | gBaseEventQueueClassInfo.clazz, "dispatchFlushCompleteEvent", "(I)V"); |
Aravind Akella | b4c76b1 | 2013-06-27 12:04:16 -0700 | [diff] [blame] | 497 | |
Peng Xu | a295835 | 2016-01-26 18:42:17 -0800 | [diff] [blame] | 498 | gBaseEventQueueClassInfo.dispatchAdditionalInfoEvent = GetMethodIDOrDie(env, |
| 499 | gBaseEventQueueClassInfo.clazz, "dispatchAdditionalInfoEvent", "(III[F[I)V"); |
| 500 | |
Mathias Agopian | db772d8 | 2013-01-31 19:31:12 -0800 | [diff] [blame] | 501 | return 0; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 502 | } |