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 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 19 | //#define LOG_NDDEBUG 0 |
| 20 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 21 | #include "JNIHelp.h" |
| 22 | #include "jni.h" |
| 23 | #include "hardware_legacy/gps.h" |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 24 | #include "hardware_legacy/gps_ni.h" |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 25 | #include "utils/Log.h" |
| 26 | #include "utils/misc.h" |
| 27 | |
| 28 | #include <string.h> |
| 29 | #include <pthread.h> |
| 30 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 31 | static pthread_mutex_t sEventMutex = PTHREAD_MUTEX_INITIALIZER; |
| 32 | static pthread_cond_t sEventCond = PTHREAD_COND_INITIALIZER; |
| 33 | static jmethodID method_reportLocation; |
| 34 | static jmethodID method_reportStatus; |
| 35 | static jmethodID method_reportSvStatus; |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 36 | static jmethodID method_reportAGpsStatus; |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 37 | static jmethodID method_reportNmea; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 38 | static jmethodID method_xtraDownloadRequest; |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 39 | static jmethodID method_reportNiNotification; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 40 | |
| 41 | static const GpsInterface* sGpsInterface = NULL; |
| 42 | static const GpsXtraInterface* sGpsXtraInterface = NULL; |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 43 | static const AGpsInterface* sAGpsInterface = NULL; |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 44 | static const GpsNiInterface* sGpsNiInterface = NULL; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 45 | |
| 46 | // data written to by GPS callbacks |
| 47 | static GpsLocation sGpsLocation; |
| 48 | static GpsStatus sGpsStatus; |
| 49 | static GpsSvStatus sGpsSvStatus; |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 50 | static AGpsStatus sAGpsStatus; |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 51 | static GpsNiNotification sGpsNiNotification; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 52 | |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 53 | // buffer for NMEA data |
| 54 | #define NMEA_SENTENCE_LENGTH 100 |
| 55 | #define NMEA_SENTENCE_COUNT 40 |
| 56 | struct NmeaSentence { |
| 57 | GpsUtcTime timestamp; |
| 58 | char nmea[NMEA_SENTENCE_LENGTH]; |
| 59 | }; |
| 60 | static NmeaSentence sNmeaBuffer[NMEA_SENTENCE_LENGTH]; |
| 61 | static int mNmeaSentenceCount = 0; |
| 62 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 63 | // a copy of the data shared by android_location_GpsLocationProvider_wait_for_event |
| 64 | // and android_location_GpsLocationProvider_read_status |
| 65 | static GpsLocation sGpsLocationCopy; |
| 66 | static GpsStatus sGpsStatusCopy; |
| 67 | static GpsSvStatus sGpsSvStatusCopy; |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 68 | static AGpsStatus sAGpsStatusCopy; |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 69 | static NmeaSentence sNmeaBufferCopy[NMEA_SENTENCE_LENGTH]; |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 70 | static GpsNiNotification sGpsNiNotificationCopy; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 71 | |
| 72 | enum CallbackType { |
| 73 | kLocation = 1, |
| 74 | kStatus = 2, |
| 75 | kSvStatus = 4, |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 76 | kAGpsStatus = 8, |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 77 | kXtraDownloadRequest = 16, |
| 78 | kDisableRequest = 32, |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 79 | kNmeaAvailable = 64, |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 80 | kNiNotification = 128, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 81 | }; |
| 82 | static int sPendingCallbacks; |
| 83 | |
| 84 | namespace android { |
| 85 | |
| 86 | static void location_callback(GpsLocation* location) |
| 87 | { |
| 88 | pthread_mutex_lock(&sEventMutex); |
| 89 | |
| 90 | sPendingCallbacks |= kLocation; |
| 91 | memcpy(&sGpsLocation, location, sizeof(sGpsLocation)); |
| 92 | |
| 93 | pthread_cond_signal(&sEventCond); |
| 94 | pthread_mutex_unlock(&sEventMutex); |
| 95 | } |
| 96 | |
| 97 | static void status_callback(GpsStatus* status) |
| 98 | { |
| 99 | pthread_mutex_lock(&sEventMutex); |
| 100 | |
| 101 | sPendingCallbacks |= kStatus; |
| 102 | memcpy(&sGpsStatus, status, sizeof(sGpsStatus)); |
| 103 | |
| 104 | pthread_cond_signal(&sEventCond); |
| 105 | pthread_mutex_unlock(&sEventMutex); |
| 106 | } |
| 107 | |
| 108 | static void sv_status_callback(GpsSvStatus* sv_status) |
| 109 | { |
| 110 | pthread_mutex_lock(&sEventMutex); |
| 111 | |
| 112 | sPendingCallbacks |= kSvStatus; |
| 113 | memcpy(&sGpsSvStatus, sv_status, sizeof(GpsSvStatus)); |
| 114 | |
| 115 | pthread_cond_signal(&sEventCond); |
| 116 | pthread_mutex_unlock(&sEventMutex); |
| 117 | } |
| 118 | |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 119 | static void nmea_callback(GpsUtcTime timestamp, const char* nmea, int length) |
| 120 | { |
| 121 | pthread_mutex_lock(&sEventMutex); |
| 122 | |
| 123 | if (length >= NMEA_SENTENCE_LENGTH) { |
| 124 | LOGE("NMEA data too long in nmea_callback (length = %d)\n", length); |
| 125 | length = NMEA_SENTENCE_LENGTH - 1; |
| 126 | } |
| 127 | if (mNmeaSentenceCount >= NMEA_SENTENCE_COUNT) { |
| 128 | LOGE("NMEA data overflowed buffer\n"); |
| 129 | pthread_mutex_unlock(&sEventMutex); |
| 130 | return; |
| 131 | } |
| 132 | |
| 133 | sPendingCallbacks |= kNmeaAvailable; |
| 134 | sNmeaBuffer[mNmeaSentenceCount].timestamp = timestamp; |
| 135 | memcpy(sNmeaBuffer[mNmeaSentenceCount].nmea, nmea, length); |
| 136 | sNmeaBuffer[mNmeaSentenceCount].nmea[length] = 0; |
| 137 | mNmeaSentenceCount++; |
| 138 | |
| 139 | pthread_cond_signal(&sEventCond); |
| 140 | pthread_mutex_unlock(&sEventMutex); |
| 141 | } |
| 142 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 143 | static void agps_status_callback(AGpsStatus* agps_status) |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 144 | { |
| 145 | pthread_mutex_lock(&sEventMutex); |
| 146 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 147 | sPendingCallbacks |= kAGpsStatus; |
| 148 | memcpy(&sAGpsStatus, agps_status, sizeof(AGpsStatus)); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 149 | |
| 150 | pthread_cond_signal(&sEventCond); |
| 151 | pthread_mutex_unlock(&sEventMutex); |
| 152 | } |
| 153 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 154 | GpsCallbacks sGpsCallbacks = { |
| 155 | location_callback, |
| 156 | status_callback, |
| 157 | sv_status_callback, |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 158 | nmea_callback |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 159 | }; |
| 160 | |
| 161 | static void |
| 162 | download_request_callback() |
| 163 | { |
| 164 | pthread_mutex_lock(&sEventMutex); |
| 165 | sPendingCallbacks |= kXtraDownloadRequest; |
| 166 | pthread_cond_signal(&sEventCond); |
| 167 | pthread_mutex_unlock(&sEventMutex); |
| 168 | } |
| 169 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 170 | static void |
| 171 | gps_ni_notify_callback(GpsNiNotification *notification) |
| 172 | { |
| 173 | LOGD("gps_ni_notify_callback: notif=%d", notification->notification_id); |
| 174 | |
| 175 | pthread_mutex_lock(&sEventMutex); |
| 176 | |
| 177 | sPendingCallbacks |= kNiNotification; |
| 178 | memcpy(&sGpsNiNotification, notification, sizeof(GpsNiNotification)); |
| 179 | |
| 180 | pthread_cond_signal(&sEventCond); |
| 181 | pthread_mutex_unlock(&sEventMutex); |
| 182 | } |
| 183 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 184 | GpsXtraCallbacks sGpsXtraCallbacks = { |
| 185 | download_request_callback, |
| 186 | }; |
| 187 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 188 | AGpsCallbacks sAGpsCallbacks = { |
| 189 | agps_status_callback, |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 190 | }; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 191 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 192 | GpsNiCallbacks sGpsNiCallbacks = { |
| 193 | gps_ni_notify_callback, |
| 194 | }; |
| 195 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 196 | static void android_location_GpsLocationProvider_class_init_native(JNIEnv* env, jclass clazz) { |
| 197 | method_reportLocation = env->GetMethodID(clazz, "reportLocation", "(IDDDFFFJ)V"); |
| 198 | method_reportStatus = env->GetMethodID(clazz, "reportStatus", "(I)V"); |
| 199 | method_reportSvStatus = env->GetMethodID(clazz, "reportSvStatus", "()V"); |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 200 | method_reportAGpsStatus = env->GetMethodID(clazz, "reportAGpsStatus", "(II)V"); |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 201 | method_reportNmea = env->GetMethodID(clazz, "reportNmea", "(IJ)V"); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 202 | method_xtraDownloadRequest = env->GetMethodID(clazz, "xtraDownloadRequest", "()V"); |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 203 | method_reportNiNotification = env->GetMethodID(clazz, "reportNiNotification", "(IIIIILjava/lang/String;Ljava/lang/String;IILjava/lang/String;)V"); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 204 | } |
| 205 | |
| 206 | static jboolean android_location_GpsLocationProvider_is_supported(JNIEnv* env, jclass clazz) { |
| 207 | if (!sGpsInterface) |
| 208 | sGpsInterface = gps_get_interface(); |
| 209 | return (sGpsInterface != NULL); |
| 210 | } |
| 211 | |
| 212 | static jboolean android_location_GpsLocationProvider_init(JNIEnv* env, jobject obj) |
| 213 | { |
| 214 | if (!sGpsInterface) |
| 215 | sGpsInterface = gps_get_interface(); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 216 | if (!sGpsInterface || sGpsInterface->init(&sGpsCallbacks) != 0) |
| 217 | return false; |
| 218 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 219 | if (!sAGpsInterface) |
| 220 | sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE); |
| 221 | if (sAGpsInterface) |
| 222 | sAGpsInterface->init(&sAGpsCallbacks); |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 223 | |
| 224 | if (!sGpsNiInterface) |
| 225 | sGpsNiInterface = (const GpsNiInterface*)sGpsInterface->get_extension(GPS_NI_INTERFACE); |
| 226 | if (sGpsNiInterface) |
| 227 | sGpsNiInterface->init(&sGpsNiCallbacks); |
| 228 | |
Mike Lockwood | c21a08f | 2009-04-30 15:16:39 -0400 | [diff] [blame] | 229 | return true; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 230 | } |
| 231 | |
| 232 | static void android_location_GpsLocationProvider_disable(JNIEnv* env, jobject obj) |
| 233 | { |
| 234 | pthread_mutex_lock(&sEventMutex); |
| 235 | sPendingCallbacks |= kDisableRequest; |
| 236 | pthread_cond_signal(&sEventCond); |
| 237 | pthread_mutex_unlock(&sEventMutex); |
| 238 | } |
| 239 | |
| 240 | static void android_location_GpsLocationProvider_cleanup(JNIEnv* env, jobject obj) |
| 241 | { |
| 242 | sGpsInterface->cleanup(); |
| 243 | } |
| 244 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 245 | static jboolean android_location_GpsLocationProvider_start(JNIEnv* env, jobject obj, jint positionMode, |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 246 | jboolean singleFix, jint fixFrequency) |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 247 | { |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 248 | int result = sGpsInterface->set_position_mode(positionMode, (singleFix ? 0 : fixFrequency)); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 249 | if (result) { |
David 'Digit' Turner | 9a940e8 | 2009-05-20 14:23:46 +0200 | [diff] [blame] | 250 | return false; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 251 | } |
| 252 | |
| 253 | return (sGpsInterface->start() == 0); |
| 254 | } |
| 255 | |
| 256 | static jboolean android_location_GpsLocationProvider_stop(JNIEnv* env, jobject obj) |
| 257 | { |
| 258 | return (sGpsInterface->stop() == 0); |
| 259 | } |
| 260 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 261 | static void android_location_GpsLocationProvider_delete_aiding_data(JNIEnv* env, jobject obj, jint flags) |
| 262 | { |
| 263 | sGpsInterface->delete_aiding_data(flags); |
| 264 | } |
| 265 | |
| 266 | static void android_location_GpsLocationProvider_wait_for_event(JNIEnv* env, jobject obj) |
| 267 | { |
| 268 | pthread_mutex_lock(&sEventMutex); |
| 269 | pthread_cond_wait(&sEventCond, &sEventMutex); |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 270 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 271 | // copy and clear the callback flags |
| 272 | int pendingCallbacks = sPendingCallbacks; |
| 273 | sPendingCallbacks = 0; |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 274 | int nmeaSentenceCount = mNmeaSentenceCount; |
| 275 | mNmeaSentenceCount = 0; |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 276 | |
| 277 | // copy everything and unlock the mutex before calling into Java code to avoid the possibility |
| 278 | // of timeouts in the GPS engine. |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 279 | if (pendingCallbacks & kLocation) |
| 280 | memcpy(&sGpsLocationCopy, &sGpsLocation, sizeof(sGpsLocationCopy)); |
| 281 | if (pendingCallbacks & kStatus) |
| 282 | memcpy(&sGpsStatusCopy, &sGpsStatus, sizeof(sGpsStatusCopy)); |
| 283 | if (pendingCallbacks & kSvStatus) |
| 284 | memcpy(&sGpsSvStatusCopy, &sGpsSvStatus, sizeof(sGpsSvStatusCopy)); |
| 285 | if (pendingCallbacks & kAGpsStatus) |
| 286 | memcpy(&sAGpsStatusCopy, &sAGpsStatus, sizeof(sAGpsStatusCopy)); |
| 287 | if (pendingCallbacks & kNmeaAvailable) |
| 288 | memcpy(&sNmeaBufferCopy, &sNmeaBuffer, nmeaSentenceCount * sizeof(sNmeaBuffer[0])); |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 289 | if (pendingCallbacks & kNiNotification) |
| 290 | memcpy(&sGpsNiNotificationCopy, &sGpsNiNotification, sizeof(sGpsNiNotificationCopy)); |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 291 | pthread_mutex_unlock(&sEventMutex); |
| 292 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 293 | if (pendingCallbacks & kLocation) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 294 | env->CallVoidMethod(obj, method_reportLocation, sGpsLocationCopy.flags, |
| 295 | (jdouble)sGpsLocationCopy.latitude, (jdouble)sGpsLocationCopy.longitude, |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 296 | (jdouble)sGpsLocationCopy.altitude, |
| 297 | (jfloat)sGpsLocationCopy.speed, (jfloat)sGpsLocationCopy.bearing, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 298 | (jfloat)sGpsLocationCopy.accuracy, (jlong)sGpsLocationCopy.timestamp); |
| 299 | } |
| 300 | if (pendingCallbacks & kStatus) { |
| 301 | env->CallVoidMethod(obj, method_reportStatus, sGpsStatusCopy.status); |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 302 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 303 | if (pendingCallbacks & kSvStatus) { |
| 304 | env->CallVoidMethod(obj, method_reportSvStatus); |
| 305 | } |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 306 | if (pendingCallbacks & kAGpsStatus) { |
| 307 | env->CallVoidMethod(obj, method_reportAGpsStatus, sAGpsStatusCopy.type, sAGpsStatusCopy.status); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 308 | } |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 309 | if (pendingCallbacks & kNmeaAvailable) { |
| 310 | for (int i = 0; i < nmeaSentenceCount; i++) { |
| 311 | env->CallVoidMethod(obj, method_reportNmea, i, sNmeaBuffer[i].timestamp); |
| 312 | } |
| 313 | } |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 314 | if (pendingCallbacks & kXtraDownloadRequest) { |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 315 | env->CallVoidMethod(obj, method_xtraDownloadRequest); |
| 316 | } |
| 317 | if (pendingCallbacks & kDisableRequest) { |
| 318 | // don't need to do anything - we are just poking so wait_for_event will return. |
| 319 | } |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 320 | if (pendingCallbacks & kNiNotification) { |
| 321 | LOGD("android_location_GpsLocationProvider_wait_for_event: sent notification callback."); |
| 322 | jstring reqId = env->NewStringUTF(sGpsNiNotificationCopy.requestor_id); |
| 323 | jstring text = env->NewStringUTF(sGpsNiNotificationCopy.text); |
| 324 | jstring extras = env->NewStringUTF(sGpsNiNotificationCopy.extras); |
| 325 | env->CallVoidMethod(obj, method_reportNiNotification, |
| 326 | sGpsNiNotificationCopy.notification_id, |
| 327 | sGpsNiNotificationCopy.ni_type, |
| 328 | sGpsNiNotificationCopy.notify_flags, |
| 329 | sGpsNiNotificationCopy.timeout, |
| 330 | sGpsNiNotificationCopy.default_response, |
| 331 | reqId, |
| 332 | text, |
| 333 | sGpsNiNotificationCopy.requestor_id_encoding, |
| 334 | sGpsNiNotificationCopy.text_encoding, |
| 335 | extras |
| 336 | ); |
| 337 | } |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 338 | } |
| 339 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 340 | static jint android_location_GpsLocationProvider_read_sv_status(JNIEnv* env, jobject obj, |
| 341 | jintArray prnArray, jfloatArray snrArray, jfloatArray elevArray, jfloatArray azumArray, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 342 | jintArray maskArray) |
| 343 | { |
| 344 | // this should only be called from within a call to reportStatus, so we don't need to lock here |
| 345 | |
| 346 | jint* prns = env->GetIntArrayElements(prnArray, 0); |
| 347 | jfloat* snrs = env->GetFloatArrayElements(snrArray, 0); |
| 348 | jfloat* elev = env->GetFloatArrayElements(elevArray, 0); |
| 349 | jfloat* azim = env->GetFloatArrayElements(azumArray, 0); |
| 350 | jint* mask = env->GetIntArrayElements(maskArray, 0); |
| 351 | |
| 352 | int num_svs = sGpsSvStatusCopy.num_svs; |
| 353 | for (int i = 0; i < num_svs; i++) { |
| 354 | prns[i] = sGpsSvStatusCopy.sv_list[i].prn; |
| 355 | snrs[i] = sGpsSvStatusCopy.sv_list[i].snr; |
| 356 | elev[i] = sGpsSvStatusCopy.sv_list[i].elevation; |
| 357 | azim[i] = sGpsSvStatusCopy.sv_list[i].azimuth; |
| 358 | } |
| 359 | mask[0] = sGpsSvStatusCopy.ephemeris_mask; |
| 360 | mask[1] = sGpsSvStatusCopy.almanac_mask; |
| 361 | mask[2] = sGpsSvStatusCopy.used_in_fix_mask; |
| 362 | |
| 363 | env->ReleaseIntArrayElements(prnArray, prns, 0); |
| 364 | env->ReleaseFloatArrayElements(snrArray, snrs, 0); |
| 365 | env->ReleaseFloatArrayElements(elevArray, elev, 0); |
| 366 | env->ReleaseFloatArrayElements(azumArray, azim, 0); |
| 367 | env->ReleaseIntArrayElements(maskArray, mask, 0); |
| 368 | return num_svs; |
| 369 | } |
| 370 | |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 371 | static jint android_location_GpsLocationProvider_read_nmea(JNIEnv* env, jobject obj, jint index, jbyteArray nmeaArray, jint buffer_size) |
| 372 | { |
| 373 | // this should only be called from within a call to reportStatus, so we don't need to lock here |
| 374 | |
| 375 | jbyte* nmea = env->GetByteArrayElements(nmeaArray, 0); |
| 376 | |
| 377 | int length = strlen(sNmeaBuffer[index].nmea); |
| 378 | if (length > buffer_size) |
| 379 | length = buffer_size; |
| 380 | memcpy(nmea, sNmeaBuffer[index].nmea, length); |
| 381 | |
| 382 | env->ReleaseByteArrayElements(nmeaArray, nmea, 0); |
| 383 | return length; |
| 384 | } |
| 385 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 386 | static void android_location_GpsLocationProvider_inject_time(JNIEnv* env, jobject obj, jlong time, |
| 387 | jlong timeReference, jint uncertainty) |
| 388 | { |
| 389 | sGpsInterface->inject_time(time, timeReference, uncertainty); |
| 390 | } |
| 391 | |
Mike Lockwood | d26ce0d | 2009-06-11 12:25:46 -0400 | [diff] [blame] | 392 | static void android_location_GpsLocationProvider_inject_location(JNIEnv* env, jobject obj, |
| 393 | jdouble latitude, jdouble longitude, jfloat accuracy) |
| 394 | { |
| 395 | sGpsInterface->inject_location(latitude, longitude, accuracy); |
| 396 | } |
| 397 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 398 | static jboolean android_location_GpsLocationProvider_supports_xtra(JNIEnv* env, jobject obj) |
| 399 | { |
| 400 | if (!sGpsXtraInterface) { |
| 401 | sGpsXtraInterface = (const GpsXtraInterface*)sGpsInterface->get_extension(GPS_XTRA_INTERFACE); |
| 402 | if (sGpsXtraInterface) { |
| 403 | int result = sGpsXtraInterface->init(&sGpsXtraCallbacks); |
| 404 | if (result) { |
| 405 | sGpsXtraInterface = NULL; |
| 406 | } |
| 407 | } |
| 408 | } |
| 409 | |
| 410 | return (sGpsXtraInterface != NULL); |
| 411 | } |
| 412 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 413 | static void android_location_GpsLocationProvider_inject_xtra_data(JNIEnv* env, jobject obj, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 414 | jbyteArray data, jint length) |
| 415 | { |
| 416 | jbyte* bytes = env->GetByteArrayElements(data, 0); |
| 417 | sGpsXtraInterface->inject_xtra_data((char *)bytes, length); |
| 418 | env->ReleaseByteArrayElements(data, bytes, 0); |
| 419 | } |
| 420 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 421 | static void android_location_GpsLocationProvider_agps_data_conn_open(JNIEnv* env, jobject obj, jstring apn) |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 422 | { |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 423 | if (!sAGpsInterface) { |
| 424 | sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE); |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 425 | } |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 426 | if (sAGpsInterface) { |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 427 | if (apn == NULL) { |
| 428 | jniThrowException(env, "java/lang/IllegalArgumentException", NULL); |
| 429 | return; |
| 430 | } |
| 431 | const char *apnStr = env->GetStringUTFChars(apn, NULL); |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 432 | sAGpsInterface->data_conn_open(apnStr); |
The Android Open Source Project | 1059253 | 2009-03-18 17:39:46 -0700 | [diff] [blame] | 433 | env->ReleaseStringUTFChars(apn, apnStr); |
| 434 | } |
| 435 | } |
| 436 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 437 | static void android_location_GpsLocationProvider_agps_data_conn_closed(JNIEnv* env, jobject obj) |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 438 | { |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 439 | if (!sAGpsInterface) { |
| 440 | sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 441 | } |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 442 | if (sAGpsInterface) { |
| 443 | sAGpsInterface->data_conn_closed(); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 444 | } |
| 445 | } |
| 446 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 447 | static void android_location_GpsLocationProvider_agps_data_conn_failed(JNIEnv* env, jobject obj) |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 448 | { |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 449 | if (!sAGpsInterface) { |
| 450 | sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 451 | } |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 452 | if (sAGpsInterface) { |
| 453 | sAGpsInterface->data_conn_failed(); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 454 | } |
| 455 | } |
| 456 | |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 457 | static void android_location_GpsLocationProvider_set_agps_server(JNIEnv* env, jobject obj, |
Mike Lockwood | a9e5461 | 2009-06-19 14:54:42 -0400 | [diff] [blame] | 458 | jint type, jstring hostname, jint port) |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 459 | { |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 460 | if (!sAGpsInterface) { |
| 461 | sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 462 | } |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 463 | if (sAGpsInterface) { |
Mike Lockwood | a9e5461 | 2009-06-19 14:54:42 -0400 | [diff] [blame] | 464 | const char *c_hostname = env->GetStringUTFChars(hostname, NULL); |
| 465 | sAGpsInterface->set_server(type, c_hostname, port); |
| 466 | env->ReleaseStringUTFChars(hostname, c_hostname); |
Mike Lockwood | 58bda98 | 2009-04-14 16:25:07 -0400 | [diff] [blame] | 467 | } |
| 468 | } |
| 469 | |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 470 | static void android_location_GpsLocationProvider_send_ni_response(JNIEnv* env, jobject obj, |
| 471 | jint notifId, jint response) |
| 472 | { |
| 473 | if (!sGpsNiInterface) |
| 474 | sGpsNiInterface = (const GpsNiInterface*)sGpsInterface->get_extension(GPS_NI_INTERFACE); |
| 475 | if (sGpsNiInterface) { |
| 476 | sGpsNiInterface->respond(notifId, response); |
| 477 | } |
| 478 | } |
| 479 | |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 480 | static JNINativeMethod sMethods[] = { |
| 481 | /* name, signature, funcPtr */ |
| 482 | {"class_init_native", "()V", (void *)android_location_GpsLocationProvider_class_init_native}, |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 483 | {"native_is_supported", "()Z", (void*)android_location_GpsLocationProvider_is_supported}, |
| 484 | {"native_init", "()Z", (void*)android_location_GpsLocationProvider_init}, |
| 485 | {"native_disable", "()V", (void*)android_location_GpsLocationProvider_disable}, |
| 486 | {"native_cleanup", "()V", (void*)android_location_GpsLocationProvider_cleanup}, |
| 487 | {"native_start", "(IZI)Z", (void*)android_location_GpsLocationProvider_start}, |
| 488 | {"native_stop", "()Z", (void*)android_location_GpsLocationProvider_stop}, |
| 489 | {"native_delete_aiding_data", "(I)V", (void*)android_location_GpsLocationProvider_delete_aiding_data}, |
| 490 | {"native_wait_for_event", "()V", (void*)android_location_GpsLocationProvider_wait_for_event}, |
| 491 | {"native_read_sv_status", "([I[F[F[F[I)I", (void*)android_location_GpsLocationProvider_read_sv_status}, |
Mike Lockwood | b16e780 | 2009-08-06 09:26:02 -0400 | [diff] [blame] | 492 | {"native_read_nmea", "(I[BI)I", (void*)android_location_GpsLocationProvider_read_nmea}, |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 493 | {"native_inject_time", "(JJI)V", (void*)android_location_GpsLocationProvider_inject_time}, |
Mike Lockwood | d26ce0d | 2009-06-11 12:25:46 -0400 | [diff] [blame] | 494 | {"native_inject_location", "(DDF)V", (void*)android_location_GpsLocationProvider_inject_location}, |
Mike Lockwood | e3635c9 | 2009-05-11 08:38:02 -0400 | [diff] [blame] | 495 | {"native_supports_xtra", "()Z", (void*)android_location_GpsLocationProvider_supports_xtra}, |
| 496 | {"native_inject_xtra_data", "([BI)V", (void*)android_location_GpsLocationProvider_inject_xtra_data}, |
| 497 | {"native_agps_data_conn_open", "(Ljava/lang/String;)V", (void*)android_location_GpsLocationProvider_agps_data_conn_open}, |
| 498 | {"native_agps_data_conn_closed", "()V", (void*)android_location_GpsLocationProvider_agps_data_conn_closed}, |
| 499 | {"native_agps_data_conn_failed", "()V", (void*)android_location_GpsLocationProvider_agps_data_conn_failed}, |
Mike Lockwood | a9e5461 | 2009-06-19 14:54:42 -0400 | [diff] [blame] | 500 | {"native_set_agps_server", "(ILjava/lang/String;I)V", (void*)android_location_GpsLocationProvider_set_agps_server}, |
Danke Xie | 22d1f9f | 2009-08-18 18:28:45 -0400 | [diff] [blame^] | 501 | {"native_send_ni_response", "(II)V", (void*)android_location_GpsLocationProvider_send_ni_response}, |
The Android Open Source Project | 9066cfe | 2009-03-03 19:31:44 -0800 | [diff] [blame] | 502 | }; |
| 503 | |
| 504 | int register_android_location_GpsLocationProvider(JNIEnv* env) |
| 505 | { |
| 506 | return jniRegisterNativeMethods(env, "com/android/internal/location/GpsLocationProvider", sMethods, NELEM(sMethods)); |
| 507 | } |
| 508 | |
| 509 | } /* namespace android */ |