The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame^] | 1 | /* |
| 2 | * Copyright (C) 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 | */ |
| 16 | |
| 17 | #define LOG_TAG "GpsLocationProvider" |
| 18 | |
| 19 | #include "JNIHelp.h" |
| 20 | #include "jni.h" |
| 21 | #include "hardware_legacy/gps.h" |
| 22 | #include "utils/Log.h" |
| 23 | #include "utils/misc.h" |
| 24 | |
| 25 | #include <string.h> |
| 26 | #include <pthread.h> |
| 27 | |
| 28 | |
| 29 | static pthread_mutex_t sEventMutex = PTHREAD_MUTEX_INITIALIZER; |
| 30 | static pthread_cond_t sEventCond = PTHREAD_COND_INITIALIZER; |
| 31 | static jmethodID method_reportLocation; |
| 32 | static jmethodID method_reportStatus; |
| 33 | static jmethodID method_reportSvStatus; |
| 34 | static jmethodID method_xtraDownloadRequest; |
| 35 | |
| 36 | static const GpsInterface* sGpsInterface = NULL; |
| 37 | static const GpsXtraInterface* sGpsXtraInterface = NULL; |
| 38 | |
| 39 | // data written to by GPS callbacks |
| 40 | static GpsLocation sGpsLocation; |
| 41 | static GpsStatus sGpsStatus; |
| 42 | static GpsSvStatus sGpsSvStatus; |
| 43 | |
| 44 | // a copy of the data shared by android_location_GpsLocationProvider_wait_for_event |
| 45 | // and android_location_GpsLocationProvider_read_status |
| 46 | static GpsLocation sGpsLocationCopy; |
| 47 | static GpsStatus sGpsStatusCopy; |
| 48 | static GpsSvStatus sGpsSvStatusCopy; |
| 49 | |
| 50 | enum CallbackType { |
| 51 | kLocation = 1, |
| 52 | kStatus = 2, |
| 53 | kSvStatus = 4, |
| 54 | kXtraDownloadRequest = 8, |
| 55 | kDisableRequest = 16, |
| 56 | }; |
| 57 | static int sPendingCallbacks; |
| 58 | |
| 59 | namespace android { |
| 60 | |
| 61 | static void location_callback(GpsLocation* location) |
| 62 | { |
| 63 | pthread_mutex_lock(&sEventMutex); |
| 64 | |
| 65 | sPendingCallbacks |= kLocation; |
| 66 | memcpy(&sGpsLocation, location, sizeof(sGpsLocation)); |
| 67 | |
| 68 | pthread_cond_signal(&sEventCond); |
| 69 | pthread_mutex_unlock(&sEventMutex); |
| 70 | } |
| 71 | |
| 72 | static void status_callback(GpsStatus* status) |
| 73 | { |
| 74 | pthread_mutex_lock(&sEventMutex); |
| 75 | |
| 76 | sPendingCallbacks |= kStatus; |
| 77 | memcpy(&sGpsStatus, status, sizeof(sGpsStatus)); |
| 78 | |
| 79 | pthread_cond_signal(&sEventCond); |
| 80 | pthread_mutex_unlock(&sEventMutex); |
| 81 | } |
| 82 | |
| 83 | static void sv_status_callback(GpsSvStatus* sv_status) |
| 84 | { |
| 85 | pthread_mutex_lock(&sEventMutex); |
| 86 | |
| 87 | sPendingCallbacks |= kSvStatus; |
| 88 | memcpy(&sGpsSvStatus, sv_status, sizeof(GpsSvStatus)); |
| 89 | |
| 90 | pthread_cond_signal(&sEventCond); |
| 91 | pthread_mutex_unlock(&sEventMutex); |
| 92 | } |
| 93 | |
| 94 | GpsCallbacks sGpsCallbacks = { |
| 95 | location_callback, |
| 96 | status_callback, |
| 97 | sv_status_callback, |
| 98 | }; |
| 99 | |
| 100 | static void |
| 101 | download_request_callback() |
| 102 | { |
| 103 | pthread_mutex_lock(&sEventMutex); |
| 104 | sPendingCallbacks |= kXtraDownloadRequest; |
| 105 | pthread_cond_signal(&sEventCond); |
| 106 | pthread_mutex_unlock(&sEventMutex); |
| 107 | } |
| 108 | |
| 109 | GpsXtraCallbacks sGpsXtraCallbacks = { |
| 110 | download_request_callback, |
| 111 | }; |
| 112 | |
| 113 | |
| 114 | static void android_location_GpsLocationProvider_class_init_native(JNIEnv* env, jclass clazz) { |
| 115 | method_reportLocation = env->GetMethodID(clazz, "reportLocation", "(IDDDFFFJ)V"); |
| 116 | method_reportStatus = env->GetMethodID(clazz, "reportStatus", "(I)V"); |
| 117 | method_reportSvStatus = env->GetMethodID(clazz, "reportSvStatus", "()V"); |
| 118 | method_xtraDownloadRequest = env->GetMethodID(clazz, "xtraDownloadRequest", "()V"); |
| 119 | } |
| 120 | |
| 121 | static jboolean android_location_GpsLocationProvider_is_supported(JNIEnv* env, jclass clazz) { |
| 122 | if (!sGpsInterface) |
| 123 | sGpsInterface = gps_get_interface(); |
| 124 | return (sGpsInterface != NULL); |
| 125 | } |
| 126 | |
| 127 | static jboolean android_location_GpsLocationProvider_init(JNIEnv* env, jobject obj) |
| 128 | { |
| 129 | if (!sGpsInterface) |
| 130 | sGpsInterface = gps_get_interface(); |
| 131 | return (sGpsInterface && sGpsInterface->init(&sGpsCallbacks) == 0); |
| 132 | } |
| 133 | |
| 134 | static void android_location_GpsLocationProvider_disable(JNIEnv* env, jobject obj) |
| 135 | { |
| 136 | pthread_mutex_lock(&sEventMutex); |
| 137 | sPendingCallbacks |= kDisableRequest; |
| 138 | pthread_cond_signal(&sEventCond); |
| 139 | pthread_mutex_unlock(&sEventMutex); |
| 140 | } |
| 141 | |
| 142 | static void android_location_GpsLocationProvider_cleanup(JNIEnv* env, jobject obj) |
| 143 | { |
| 144 | sGpsInterface->cleanup(); |
| 145 | } |
| 146 | |
| 147 | static jboolean android_location_GpsLocationProvider_start(JNIEnv* env, jobject obj, jboolean singleFix, jint fixFrequency) |
| 148 | { |
| 149 | int result = sGpsInterface->set_position_mode(GPS_POSITION_MODE_STANDALONE, (singleFix ? 0 : fixFrequency)); |
| 150 | if (result) { |
| 151 | return result; |
| 152 | } |
| 153 | |
| 154 | return (sGpsInterface->start() == 0); |
| 155 | } |
| 156 | |
| 157 | static jboolean android_location_GpsLocationProvider_stop(JNIEnv* env, jobject obj) |
| 158 | { |
| 159 | return (sGpsInterface->stop() == 0); |
| 160 | } |
| 161 | |
| 162 | static void android_location_GpsLocationProvider_set_fix_frequency(JNIEnv* env, jobject obj, jint fixFrequency) |
| 163 | { |
| 164 | if (sGpsInterface->set_fix_frequency) |
| 165 | sGpsInterface->set_fix_frequency(fixFrequency); |
| 166 | } |
| 167 | |
| 168 | static void android_location_GpsLocationProvider_delete_aiding_data(JNIEnv* env, jobject obj, jint flags) |
| 169 | { |
| 170 | sGpsInterface->delete_aiding_data(flags); |
| 171 | } |
| 172 | |
| 173 | static void android_location_GpsLocationProvider_wait_for_event(JNIEnv* env, jobject obj) |
| 174 | { |
| 175 | pthread_mutex_lock(&sEventMutex); |
| 176 | pthread_cond_wait(&sEventCond, &sEventMutex); |
| 177 | |
| 178 | // copy and clear the callback flags |
| 179 | int pendingCallbacks = sPendingCallbacks; |
| 180 | sPendingCallbacks = 0; |
| 181 | |
| 182 | // copy everything and unlock the mutex before calling into Java code to avoid the possibility |
| 183 | // of timeouts in the GPS engine. |
| 184 | memcpy(&sGpsLocationCopy, &sGpsLocation, sizeof(sGpsLocationCopy)); |
| 185 | memcpy(&sGpsStatusCopy, &sGpsStatus, sizeof(sGpsStatusCopy)); |
| 186 | memcpy(&sGpsSvStatusCopy, &sGpsSvStatus, sizeof(sGpsSvStatusCopy)); |
| 187 | pthread_mutex_unlock(&sEventMutex); |
| 188 | |
| 189 | if (pendingCallbacks & kLocation) { |
| 190 | env->CallVoidMethod(obj, method_reportLocation, sGpsLocationCopy.flags, |
| 191 | (jdouble)sGpsLocationCopy.latitude, (jdouble)sGpsLocationCopy.longitude, |
| 192 | (jdouble)sGpsLocationCopy.altitude, |
| 193 | (jfloat)sGpsLocationCopy.speed, (jfloat)sGpsLocationCopy.bearing, |
| 194 | (jfloat)sGpsLocationCopy.accuracy, (jlong)sGpsLocationCopy.timestamp); |
| 195 | } |
| 196 | if (pendingCallbacks & kStatus) { |
| 197 | env->CallVoidMethod(obj, method_reportStatus, sGpsStatusCopy.status); |
| 198 | } |
| 199 | if (pendingCallbacks & kSvStatus) { |
| 200 | env->CallVoidMethod(obj, method_reportSvStatus); |
| 201 | } |
| 202 | if (pendingCallbacks & kXtraDownloadRequest) { |
| 203 | env->CallVoidMethod(obj, method_xtraDownloadRequest); |
| 204 | } |
| 205 | if (pendingCallbacks & kDisableRequest) { |
| 206 | // don't need to do anything - we are just poking so wait_for_event will return. |
| 207 | } |
| 208 | } |
| 209 | |
| 210 | static jint android_location_GpsLocationProvider_read_sv_status(JNIEnv* env, jobject obj, |
| 211 | jintArray prnArray, jfloatArray snrArray, jfloatArray elevArray, jfloatArray azumArray, |
| 212 | jintArray maskArray) |
| 213 | { |
| 214 | // this should only be called from within a call to reportStatus, so we don't need to lock here |
| 215 | |
| 216 | jint* prns = env->GetIntArrayElements(prnArray, 0); |
| 217 | jfloat* snrs = env->GetFloatArrayElements(snrArray, 0); |
| 218 | jfloat* elev = env->GetFloatArrayElements(elevArray, 0); |
| 219 | jfloat* azim = env->GetFloatArrayElements(azumArray, 0); |
| 220 | jint* mask = env->GetIntArrayElements(maskArray, 0); |
| 221 | |
| 222 | int num_svs = sGpsSvStatusCopy.num_svs; |
| 223 | for (int i = 0; i < num_svs; i++) { |
| 224 | prns[i] = sGpsSvStatusCopy.sv_list[i].prn; |
| 225 | snrs[i] = sGpsSvStatusCopy.sv_list[i].snr; |
| 226 | elev[i] = sGpsSvStatusCopy.sv_list[i].elevation; |
| 227 | azim[i] = sGpsSvStatusCopy.sv_list[i].azimuth; |
| 228 | } |
| 229 | mask[0] = sGpsSvStatusCopy.ephemeris_mask; |
| 230 | mask[1] = sGpsSvStatusCopy.almanac_mask; |
| 231 | mask[2] = sGpsSvStatusCopy.used_in_fix_mask; |
| 232 | |
| 233 | env->ReleaseIntArrayElements(prnArray, prns, 0); |
| 234 | env->ReleaseFloatArrayElements(snrArray, snrs, 0); |
| 235 | env->ReleaseFloatArrayElements(elevArray, elev, 0); |
| 236 | env->ReleaseFloatArrayElements(azumArray, azim, 0); |
| 237 | env->ReleaseIntArrayElements(maskArray, mask, 0); |
| 238 | return num_svs; |
| 239 | } |
| 240 | |
| 241 | static void android_location_GpsLocationProvider_inject_time(JNIEnv* env, jobject obj, jlong time, |
| 242 | jlong timeReference, jint uncertainty) |
| 243 | { |
| 244 | sGpsInterface->inject_time(time, timeReference, uncertainty); |
| 245 | } |
| 246 | |
| 247 | static jboolean android_location_GpsLocationProvider_supports_xtra(JNIEnv* env, jobject obj) |
| 248 | { |
| 249 | if (!sGpsXtraInterface) { |
| 250 | sGpsXtraInterface = (const GpsXtraInterface*)sGpsInterface->get_extension(GPS_XTRA_INTERFACE); |
| 251 | if (sGpsXtraInterface) { |
| 252 | int result = sGpsXtraInterface->init(&sGpsXtraCallbacks); |
| 253 | if (result) { |
| 254 | sGpsXtraInterface = NULL; |
| 255 | } |
| 256 | } |
| 257 | } |
| 258 | |
| 259 | return (sGpsXtraInterface != NULL); |
| 260 | } |
| 261 | |
| 262 | static void android_location_GpsLocationProvider_inject_xtra_data(JNIEnv* env, jobject obj, |
| 263 | jbyteArray data, jint length) |
| 264 | { |
| 265 | jbyte* bytes = env->GetByteArrayElements(data, 0); |
| 266 | sGpsXtraInterface->inject_xtra_data((char *)bytes, length); |
| 267 | env->ReleaseByteArrayElements(data, bytes, 0); |
| 268 | } |
| 269 | |
| 270 | static JNINativeMethod sMethods[] = { |
| 271 | /* name, signature, funcPtr */ |
| 272 | {"class_init_native", "()V", (void *)android_location_GpsLocationProvider_class_init_native}, |
| 273 | {"native_is_supported", "()Z", (void*)android_location_GpsLocationProvider_is_supported}, |
| 274 | {"native_init", "()Z", (void*)android_location_GpsLocationProvider_init}, |
| 275 | {"native_disable", "()V", (void*)android_location_GpsLocationProvider_disable}, |
| 276 | {"native_cleanup", "()V", (void*)android_location_GpsLocationProvider_cleanup}, |
| 277 | {"native_start", "(ZI)Z", (void*)android_location_GpsLocationProvider_start}, |
| 278 | {"native_stop", "()Z", (void*)android_location_GpsLocationProvider_stop}, |
| 279 | {"native_set_fix_frequency", "(I)V", (void*)android_location_GpsLocationProvider_set_fix_frequency}, |
| 280 | {"native_delete_aiding_data", "(I)V", (void*)android_location_GpsLocationProvider_delete_aiding_data}, |
| 281 | {"native_wait_for_event", "()V", (void*)android_location_GpsLocationProvider_wait_for_event}, |
| 282 | {"native_read_sv_status", "([I[F[F[F[I)I", (void*)android_location_GpsLocationProvider_read_sv_status}, |
| 283 | {"native_inject_time", "(JJI)V", (void*)android_location_GpsLocationProvider_inject_time}, |
| 284 | {"native_supports_xtra", "()Z", (void*)android_location_GpsLocationProvider_supports_xtra}, |
| 285 | {"native_inject_xtra_data", "([BI)V", (void*)android_location_GpsLocationProvider_inject_xtra_data}, |
| 286 | }; |
| 287 | |
| 288 | int register_android_location_GpsLocationProvider(JNIEnv* env) |
| 289 | { |
| 290 | return jniRegisterNativeMethods(env, "com/android/internal/location/GpsLocationProvider", sMethods, NELEM(sMethods)); |
| 291 | } |
| 292 | |
| 293 | } /* namespace android */ |