blob: afd6ca89121f320d01e78533a3f02cca90d6b12d [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
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 Xie22d1f9f2009-08-18 18:28:45 -040019//#define LOG_NDDEBUG 0
20
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080021#include "JNIHelp.h"
22#include "jni.h"
Mike Lockwoodb7ff4572010-04-05 15:24:34 -040023#include "hardware/hardware.h"
24#include "hardware/gps.h"
Mike Lockwood8f5a8002010-04-07 09:05:26 -040025#include "hardware_legacy/power.h"
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080026#include "utils/Log.h"
27#include "utils/misc.h"
28
29#include <string.h>
30#include <pthread.h>
31
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080032static pthread_mutex_t sEventMutex = PTHREAD_MUTEX_INITIALIZER;
33static pthread_cond_t sEventCond = PTHREAD_COND_INITIALIZER;
34static jmethodID method_reportLocation;
35static jmethodID method_reportStatus;
36static jmethodID method_reportSvStatus;
Mike Lockwoode3635c92009-05-11 08:38:02 -040037static jmethodID method_reportAGpsStatus;
Mike Lockwoodb16e7802009-08-06 09:26:02 -040038static jmethodID method_reportNmea;
Mike Lockwood04598b62010-04-14 17:17:24 -040039static jmethodID method_setEngineCapabilities;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080040static jmethodID method_xtraDownloadRequest;
Danke Xie22d1f9f2009-08-18 18:28:45 -040041static jmethodID method_reportNiNotification;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080042
43static const GpsInterface* sGpsInterface = NULL;
44static const GpsXtraInterface* sGpsXtraInterface = NULL;
Mike Lockwoode3635c92009-05-11 08:38:02 -040045static const AGpsInterface* sAGpsInterface = NULL;
Danke Xie22d1f9f2009-08-18 18:28:45 -040046static const GpsNiInterface* sGpsNiInterface = NULL;
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -060047static const GpsDebugInterface* sGpsDebugInterface = NULL;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080048
49// data written to by GPS callbacks
50static GpsLocation sGpsLocation;
51static GpsStatus sGpsStatus;
52static GpsSvStatus sGpsSvStatus;
Mike Lockwoode3635c92009-05-11 08:38:02 -040053static AGpsStatus sAGpsStatus;
Danke Xie22d1f9f2009-08-18 18:28:45 -040054static GpsNiNotification sGpsNiNotification;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080055
Mike Lockwood8f5a8002010-04-07 09:05:26 -040056#define WAKE_LOCK_NAME "GPS"
57
Mike Lockwoodb16e7802009-08-06 09:26:02 -040058// buffer for NMEA data
59#define NMEA_SENTENCE_LENGTH 100
60#define NMEA_SENTENCE_COUNT 40
61struct NmeaSentence {
62 GpsUtcTime timestamp;
63 char nmea[NMEA_SENTENCE_LENGTH];
64};
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -060065static NmeaSentence sNmeaBuffer[NMEA_SENTENCE_COUNT];
Mike Lockwoodb16e7802009-08-06 09:26:02 -040066static int mNmeaSentenceCount = 0;
67
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080068// a copy of the data shared by android_location_GpsLocationProvider_wait_for_event
69// and android_location_GpsLocationProvider_read_status
70static GpsLocation sGpsLocationCopy;
71static GpsStatus sGpsStatusCopy;
72static GpsSvStatus sGpsSvStatusCopy;
Mike Lockwoode3635c92009-05-11 08:38:02 -040073static AGpsStatus sAGpsStatusCopy;
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -060074static NmeaSentence sNmeaBufferCopy[NMEA_SENTENCE_COUNT];
Danke Xie22d1f9f2009-08-18 18:28:45 -040075static GpsNiNotification sGpsNiNotificationCopy;
Mike Lockwood04598b62010-04-14 17:17:24 -040076static uint32_t sEngineCapabilities;
77
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080078
79enum CallbackType {
80 kLocation = 1,
81 kStatus = 2,
82 kSvStatus = 4,
Mike Lockwoode3635c92009-05-11 08:38:02 -040083 kAGpsStatus = 8,
Mike Lockwood58bda982009-04-14 16:25:07 -040084 kXtraDownloadRequest = 16,
85 kDisableRequest = 32,
Mike Lockwoodb16e7802009-08-06 09:26:02 -040086 kNmeaAvailable = 64,
Danke Xie22d1f9f2009-08-18 18:28:45 -040087 kNiNotification = 128,
Mike Lockwood04598b62010-04-14 17:17:24 -040088 kSetCapabilities = 256,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080089};
90static int sPendingCallbacks;
91
92namespace android {
93
94static void location_callback(GpsLocation* location)
95{
96 pthread_mutex_lock(&sEventMutex);
97
98 sPendingCallbacks |= kLocation;
99 memcpy(&sGpsLocation, location, sizeof(sGpsLocation));
100
101 pthread_cond_signal(&sEventCond);
102 pthread_mutex_unlock(&sEventMutex);
103}
104
105static void status_callback(GpsStatus* status)
106{
107 pthread_mutex_lock(&sEventMutex);
108
109 sPendingCallbacks |= kStatus;
110 memcpy(&sGpsStatus, status, sizeof(sGpsStatus));
111
112 pthread_cond_signal(&sEventCond);
113 pthread_mutex_unlock(&sEventMutex);
114}
115
116static void sv_status_callback(GpsSvStatus* sv_status)
117{
118 pthread_mutex_lock(&sEventMutex);
119
120 sPendingCallbacks |= kSvStatus;
121 memcpy(&sGpsSvStatus, sv_status, sizeof(GpsSvStatus));
122
123 pthread_cond_signal(&sEventCond);
124 pthread_mutex_unlock(&sEventMutex);
125}
126
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400127static void nmea_callback(GpsUtcTime timestamp, const char* nmea, int length)
128{
129 pthread_mutex_lock(&sEventMutex);
130
131 if (length >= NMEA_SENTENCE_LENGTH) {
132 LOGE("NMEA data too long in nmea_callback (length = %d)\n", length);
133 length = NMEA_SENTENCE_LENGTH - 1;
134 }
135 if (mNmeaSentenceCount >= NMEA_SENTENCE_COUNT) {
136 LOGE("NMEA data overflowed buffer\n");
137 pthread_mutex_unlock(&sEventMutex);
138 return;
139 }
140
141 sPendingCallbacks |= kNmeaAvailable;
142 sNmeaBuffer[mNmeaSentenceCount].timestamp = timestamp;
143 memcpy(sNmeaBuffer[mNmeaSentenceCount].nmea, nmea, length);
144 sNmeaBuffer[mNmeaSentenceCount].nmea[length] = 0;
145 mNmeaSentenceCount++;
146
147 pthread_cond_signal(&sEventCond);
148 pthread_mutex_unlock(&sEventMutex);
149}
150
Mike Lockwood04598b62010-04-14 17:17:24 -0400151static void set_capabilities_callback(uint32_t capabilities)
152{
153 LOGD("set_capabilities_callback: %08X", capabilities);
154
155 pthread_mutex_lock(&sEventMutex);
156
157 sPendingCallbacks |= kSetCapabilities;
158 sEngineCapabilities = capabilities;
159
160 pthread_cond_signal(&sEventCond);
161 pthread_mutex_unlock(&sEventMutex);
162}
163
Mike Lockwood8f5a8002010-04-07 09:05:26 -0400164static void acquire_wakelock_callback()
165{
166 acquire_wake_lock(PARTIAL_WAKE_LOCK, WAKE_LOCK_NAME);
167}
168
169static void release_wakelock_callback()
170{
171 release_wake_lock(WAKE_LOCK_NAME);
172}
173
Mike Lockwoode3635c92009-05-11 08:38:02 -0400174static void agps_status_callback(AGpsStatus* agps_status)
Mike Lockwood58bda982009-04-14 16:25:07 -0400175{
176 pthread_mutex_lock(&sEventMutex);
177
Mike Lockwoode3635c92009-05-11 08:38:02 -0400178 sPendingCallbacks |= kAGpsStatus;
179 memcpy(&sAGpsStatus, agps_status, sizeof(AGpsStatus));
Mike Lockwood58bda982009-04-14 16:25:07 -0400180
181 pthread_cond_signal(&sEventCond);
182 pthread_mutex_unlock(&sEventMutex);
183}
184
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800185GpsCallbacks sGpsCallbacks = {
Mike Lockwood8f5a8002010-04-07 09:05:26 -0400186 sizeof(GpsCallbacks),
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800187 location_callback,
188 status_callback,
189 sv_status_callback,
Mike Lockwood8f5a8002010-04-07 09:05:26 -0400190 nmea_callback,
Mike Lockwood04598b62010-04-14 17:17:24 -0400191 set_capabilities_callback,
Mike Lockwood8f5a8002010-04-07 09:05:26 -0400192 acquire_wakelock_callback,
193 release_wakelock_callback,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800194};
195
196static void
197download_request_callback()
198{
199 pthread_mutex_lock(&sEventMutex);
200 sPendingCallbacks |= kXtraDownloadRequest;
201 pthread_cond_signal(&sEventCond);
202 pthread_mutex_unlock(&sEventMutex);
203}
204
Danke Xie22d1f9f2009-08-18 18:28:45 -0400205static void
206gps_ni_notify_callback(GpsNiNotification *notification)
207{
208 LOGD("gps_ni_notify_callback: notif=%d", notification->notification_id);
209
210 pthread_mutex_lock(&sEventMutex);
211
212 sPendingCallbacks |= kNiNotification;
213 memcpy(&sGpsNiNotification, notification, sizeof(GpsNiNotification));
214
215 pthread_cond_signal(&sEventCond);
216 pthread_mutex_unlock(&sEventMutex);
217}
218
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800219GpsXtraCallbacks sGpsXtraCallbacks = {
220 download_request_callback,
221};
222
Mike Lockwoode3635c92009-05-11 08:38:02 -0400223AGpsCallbacks sAGpsCallbacks = {
224 agps_status_callback,
Mike Lockwood58bda982009-04-14 16:25:07 -0400225};
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800226
Danke Xie22d1f9f2009-08-18 18:28:45 -0400227GpsNiCallbacks sGpsNiCallbacks = {
228 gps_ni_notify_callback,
229};
230
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800231static void android_location_GpsLocationProvider_class_init_native(JNIEnv* env, jclass clazz) {
232 method_reportLocation = env->GetMethodID(clazz, "reportLocation", "(IDDDFFFJ)V");
233 method_reportStatus = env->GetMethodID(clazz, "reportStatus", "(I)V");
234 method_reportSvStatus = env->GetMethodID(clazz, "reportSvStatus", "()V");
Mike Lockwoode3635c92009-05-11 08:38:02 -0400235 method_reportAGpsStatus = env->GetMethodID(clazz, "reportAGpsStatus", "(II)V");
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400236 method_reportNmea = env->GetMethodID(clazz, "reportNmea", "(IJ)V");
Mike Lockwood04598b62010-04-14 17:17:24 -0400237 method_setEngineCapabilities = env->GetMethodID(clazz, "setEngineCapabilities", "(I)V");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800238 method_xtraDownloadRequest = env->GetMethodID(clazz, "xtraDownloadRequest", "()V");
Danke Xie22d1f9f2009-08-18 18:28:45 -0400239 method_reportNiNotification = env->GetMethodID(clazz, "reportNiNotification", "(IIIIILjava/lang/String;Ljava/lang/String;IILjava/lang/String;)V");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800240}
241
Mike Lockwoodb7ff4572010-04-05 15:24:34 -0400242static const GpsInterface* get_gps_interface() {
243 int err;
244 hw_module_t* module;
245 const GpsInterface* interface = NULL;
246
247 err = hw_get_module(GPS_HARDWARE_MODULE_ID, (hw_module_t const**)&module);
248 if (err == 0) {
249 hw_device_t* device;
250 err = module->methods->open(module, GPS_HARDWARE_MODULE_ID, &device);
251 if (err == 0) {
252 gps_device_t* gps_device = (gps_device_t *)device;
253 interface = gps_device->get_gps_interface(gps_device);
254 }
255 }
256
257 return interface;
258}
259
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800260static jboolean android_location_GpsLocationProvider_is_supported(JNIEnv* env, jclass clazz) {
261 if (!sGpsInterface)
Mike Lockwoodb7ff4572010-04-05 15:24:34 -0400262 sGpsInterface = get_gps_interface();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800263 return (sGpsInterface != NULL);
264}
265
266static jboolean android_location_GpsLocationProvider_init(JNIEnv* env, jobject obj)
267{
268 if (!sGpsInterface)
Mike Lockwoodb7ff4572010-04-05 15:24:34 -0400269 sGpsInterface = get_gps_interface();
Mike Lockwood58bda982009-04-14 16:25:07 -0400270 if (!sGpsInterface || sGpsInterface->init(&sGpsCallbacks) != 0)
271 return false;
272
Mike Lockwoode3635c92009-05-11 08:38:02 -0400273 if (!sAGpsInterface)
274 sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE);
275 if (sAGpsInterface)
276 sAGpsInterface->init(&sAGpsCallbacks);
Danke Xie22d1f9f2009-08-18 18:28:45 -0400277
278 if (!sGpsNiInterface)
279 sGpsNiInterface = (const GpsNiInterface*)sGpsInterface->get_extension(GPS_NI_INTERFACE);
280 if (sGpsNiInterface)
281 sGpsNiInterface->init(&sGpsNiCallbacks);
282
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -0600283 if (!sGpsDebugInterface)
284 sGpsDebugInterface = (const GpsDebugInterface*)sGpsInterface->get_extension(GPS_DEBUG_INTERFACE);
285
Mike Lockwoodc21a08f2009-04-30 15:16:39 -0400286 return true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800287}
288
289static void android_location_GpsLocationProvider_disable(JNIEnv* env, jobject obj)
290{
291 pthread_mutex_lock(&sEventMutex);
292 sPendingCallbacks |= kDisableRequest;
293 pthread_cond_signal(&sEventCond);
294 pthread_mutex_unlock(&sEventMutex);
295}
296
297static void android_location_GpsLocationProvider_cleanup(JNIEnv* env, jobject obj)
298{
299 sGpsInterface->cleanup();
300}
301
Mike Lockwood04598b62010-04-14 17:17:24 -0400302static jboolean android_location_GpsLocationProvider_set_position_mode(JNIEnv* env, jobject obj,
303 jint mode, jint recurrence, jint min_interval, jint preferred_accuracy, jint preferred_time)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800304{
Mike Lockwood04598b62010-04-14 17:17:24 -0400305 return (sGpsInterface->set_position_mode(mode, recurrence, min_interval, preferred_accuracy,
306 preferred_time) == 0);
307}
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800308
Mike Lockwood04598b62010-04-14 17:17:24 -0400309static jboolean android_location_GpsLocationProvider_start(JNIEnv* env, jobject obj)
310{
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800311 return (sGpsInterface->start() == 0);
312}
313
314static jboolean android_location_GpsLocationProvider_stop(JNIEnv* env, jobject obj)
315{
316 return (sGpsInterface->stop() == 0);
317}
318
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800319static void android_location_GpsLocationProvider_delete_aiding_data(JNIEnv* env, jobject obj, jint flags)
320{
321 sGpsInterface->delete_aiding_data(flags);
322}
323
324static void android_location_GpsLocationProvider_wait_for_event(JNIEnv* env, jobject obj)
325{
326 pthread_mutex_lock(&sEventMutex);
Jared Suttles8b1243e2009-11-04 16:53:44 -0600327 while (sPendingCallbacks == 0) {
328 pthread_cond_wait(&sEventCond, &sEventMutex);
329 }
Danke Xie22d1f9f2009-08-18 18:28:45 -0400330
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800331 // copy and clear the callback flags
332 int pendingCallbacks = sPendingCallbacks;
333 sPendingCallbacks = 0;
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400334 int nmeaSentenceCount = mNmeaSentenceCount;
335 mNmeaSentenceCount = 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800336
337 // copy everything and unlock the mutex before calling into Java code to avoid the possibility
338 // of timeouts in the GPS engine.
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400339 if (pendingCallbacks & kLocation)
340 memcpy(&sGpsLocationCopy, &sGpsLocation, sizeof(sGpsLocationCopy));
341 if (pendingCallbacks & kStatus)
342 memcpy(&sGpsStatusCopy, &sGpsStatus, sizeof(sGpsStatusCopy));
343 if (pendingCallbacks & kSvStatus)
344 memcpy(&sGpsSvStatusCopy, &sGpsSvStatus, sizeof(sGpsSvStatusCopy));
345 if (pendingCallbacks & kAGpsStatus)
346 memcpy(&sAGpsStatusCopy, &sAGpsStatus, sizeof(sAGpsStatusCopy));
347 if (pendingCallbacks & kNmeaAvailable)
348 memcpy(&sNmeaBufferCopy, &sNmeaBuffer, nmeaSentenceCount * sizeof(sNmeaBuffer[0]));
Danke Xie22d1f9f2009-08-18 18:28:45 -0400349 if (pendingCallbacks & kNiNotification)
350 memcpy(&sGpsNiNotificationCopy, &sGpsNiNotification, sizeof(sGpsNiNotificationCopy));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800351 pthread_mutex_unlock(&sEventMutex);
352
Danke Xie22d1f9f2009-08-18 18:28:45 -0400353 if (pendingCallbacks & kLocation) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800354 env->CallVoidMethod(obj, method_reportLocation, sGpsLocationCopy.flags,
355 (jdouble)sGpsLocationCopy.latitude, (jdouble)sGpsLocationCopy.longitude,
Danke Xie22d1f9f2009-08-18 18:28:45 -0400356 (jdouble)sGpsLocationCopy.altitude,
357 (jfloat)sGpsLocationCopy.speed, (jfloat)sGpsLocationCopy.bearing,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800358 (jfloat)sGpsLocationCopy.accuracy, (jlong)sGpsLocationCopy.timestamp);
359 }
360 if (pendingCallbacks & kStatus) {
361 env->CallVoidMethod(obj, method_reportStatus, sGpsStatusCopy.status);
Danke Xie22d1f9f2009-08-18 18:28:45 -0400362 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800363 if (pendingCallbacks & kSvStatus) {
364 env->CallVoidMethod(obj, method_reportSvStatus);
365 }
Mike Lockwoode3635c92009-05-11 08:38:02 -0400366 if (pendingCallbacks & kAGpsStatus) {
367 env->CallVoidMethod(obj, method_reportAGpsStatus, sAGpsStatusCopy.type, sAGpsStatusCopy.status);
Mike Lockwood58bda982009-04-14 16:25:07 -0400368 }
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400369 if (pendingCallbacks & kNmeaAvailable) {
370 for (int i = 0; i < nmeaSentenceCount; i++) {
371 env->CallVoidMethod(obj, method_reportNmea, i, sNmeaBuffer[i].timestamp);
372 }
373 }
Danke Xie22d1f9f2009-08-18 18:28:45 -0400374 if (pendingCallbacks & kXtraDownloadRequest) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800375 env->CallVoidMethod(obj, method_xtraDownloadRequest);
376 }
377 if (pendingCallbacks & kDisableRequest) {
378 // don't need to do anything - we are just poking so wait_for_event will return.
379 }
Danke Xie22d1f9f2009-08-18 18:28:45 -0400380 if (pendingCallbacks & kNiNotification) {
381 LOGD("android_location_GpsLocationProvider_wait_for_event: sent notification callback.");
382 jstring reqId = env->NewStringUTF(sGpsNiNotificationCopy.requestor_id);
383 jstring text = env->NewStringUTF(sGpsNiNotificationCopy.text);
384 jstring extras = env->NewStringUTF(sGpsNiNotificationCopy.extras);
385 env->CallVoidMethod(obj, method_reportNiNotification,
386 sGpsNiNotificationCopy.notification_id,
387 sGpsNiNotificationCopy.ni_type,
388 sGpsNiNotificationCopy.notify_flags,
389 sGpsNiNotificationCopy.timeout,
390 sGpsNiNotificationCopy.default_response,
391 reqId,
392 text,
393 sGpsNiNotificationCopy.requestor_id_encoding,
394 sGpsNiNotificationCopy.text_encoding,
395 extras
396 );
397 }
Mike Lockwood04598b62010-04-14 17:17:24 -0400398 if (pendingCallbacks & kSetCapabilities) {
399 env->CallVoidMethod(obj, method_setEngineCapabilities, sEngineCapabilities);
400 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800401}
402
Danke Xie22d1f9f2009-08-18 18:28:45 -0400403static jint android_location_GpsLocationProvider_read_sv_status(JNIEnv* env, jobject obj,
404 jintArray prnArray, jfloatArray snrArray, jfloatArray elevArray, jfloatArray azumArray,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800405 jintArray maskArray)
406{
407 // this should only be called from within a call to reportStatus, so we don't need to lock here
408
409 jint* prns = env->GetIntArrayElements(prnArray, 0);
410 jfloat* snrs = env->GetFloatArrayElements(snrArray, 0);
411 jfloat* elev = env->GetFloatArrayElements(elevArray, 0);
412 jfloat* azim = env->GetFloatArrayElements(azumArray, 0);
413 jint* mask = env->GetIntArrayElements(maskArray, 0);
414
415 int num_svs = sGpsSvStatusCopy.num_svs;
416 for (int i = 0; i < num_svs; i++) {
417 prns[i] = sGpsSvStatusCopy.sv_list[i].prn;
418 snrs[i] = sGpsSvStatusCopy.sv_list[i].snr;
419 elev[i] = sGpsSvStatusCopy.sv_list[i].elevation;
420 azim[i] = sGpsSvStatusCopy.sv_list[i].azimuth;
421 }
422 mask[0] = sGpsSvStatusCopy.ephemeris_mask;
423 mask[1] = sGpsSvStatusCopy.almanac_mask;
424 mask[2] = sGpsSvStatusCopy.used_in_fix_mask;
425
426 env->ReleaseIntArrayElements(prnArray, prns, 0);
427 env->ReleaseFloatArrayElements(snrArray, snrs, 0);
428 env->ReleaseFloatArrayElements(elevArray, elev, 0);
429 env->ReleaseFloatArrayElements(azumArray, azim, 0);
430 env->ReleaseIntArrayElements(maskArray, mask, 0);
431 return num_svs;
432}
433
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400434static jint android_location_GpsLocationProvider_read_nmea(JNIEnv* env, jobject obj, jint index, jbyteArray nmeaArray, jint buffer_size)
435{
Eddie Hoeb40aae2009-09-04 07:03:22 -0400436 // this should only be called from within a call to reportNmea, so we don't need to lock here
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400437
438 jbyte* nmea = env->GetByteArrayElements(nmeaArray, 0);
439
Eddie Hoeb40aae2009-09-04 07:03:22 -0400440 int length = strlen(sNmeaBufferCopy[index].nmea);
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400441 if (length > buffer_size)
442 length = buffer_size;
Eddie Hoeb40aae2009-09-04 07:03:22 -0400443 memcpy(nmea, sNmeaBufferCopy[index].nmea, length);
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400444
445 env->ReleaseByteArrayElements(nmeaArray, nmea, 0);
446 return length;
447}
448
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800449static void android_location_GpsLocationProvider_inject_time(JNIEnv* env, jobject obj, jlong time,
450 jlong timeReference, jint uncertainty)
451{
452 sGpsInterface->inject_time(time, timeReference, uncertainty);
453}
454
Mike Lockwoodd26ce0d2009-06-11 12:25:46 -0400455static void android_location_GpsLocationProvider_inject_location(JNIEnv* env, jobject obj,
456 jdouble latitude, jdouble longitude, jfloat accuracy)
457{
458 sGpsInterface->inject_location(latitude, longitude, accuracy);
459}
460
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800461static jboolean android_location_GpsLocationProvider_supports_xtra(JNIEnv* env, jobject obj)
462{
463 if (!sGpsXtraInterface) {
464 sGpsXtraInterface = (const GpsXtraInterface*)sGpsInterface->get_extension(GPS_XTRA_INTERFACE);
465 if (sGpsXtraInterface) {
466 int result = sGpsXtraInterface->init(&sGpsXtraCallbacks);
467 if (result) {
468 sGpsXtraInterface = NULL;
469 }
470 }
471 }
472
473 return (sGpsXtraInterface != NULL);
474}
475
Danke Xie22d1f9f2009-08-18 18:28:45 -0400476static void android_location_GpsLocationProvider_inject_xtra_data(JNIEnv* env, jobject obj,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800477 jbyteArray data, jint length)
478{
479 jbyte* bytes = env->GetByteArrayElements(data, 0);
480 sGpsXtraInterface->inject_xtra_data((char *)bytes, length);
481 env->ReleaseByteArrayElements(data, bytes, 0);
482}
483
Mike Lockwoode3635c92009-05-11 08:38:02 -0400484static void android_location_GpsLocationProvider_agps_data_conn_open(JNIEnv* env, jobject obj, jstring apn)
The Android Open Source Project10592532009-03-18 17:39:46 -0700485{
Mike Lockwoode3635c92009-05-11 08:38:02 -0400486 if (!sAGpsInterface) {
487 sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE);
The Android Open Source Project10592532009-03-18 17:39:46 -0700488 }
Mike Lockwoode3635c92009-05-11 08:38:02 -0400489 if (sAGpsInterface) {
The Android Open Source Project10592532009-03-18 17:39:46 -0700490 if (apn == NULL) {
491 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
492 return;
493 }
494 const char *apnStr = env->GetStringUTFChars(apn, NULL);
Mike Lockwoode3635c92009-05-11 08:38:02 -0400495 sAGpsInterface->data_conn_open(apnStr);
The Android Open Source Project10592532009-03-18 17:39:46 -0700496 env->ReleaseStringUTFChars(apn, apnStr);
497 }
498}
499
Mike Lockwoode3635c92009-05-11 08:38:02 -0400500static void android_location_GpsLocationProvider_agps_data_conn_closed(JNIEnv* env, jobject obj)
Mike Lockwood58bda982009-04-14 16:25:07 -0400501{
Mike Lockwoode3635c92009-05-11 08:38:02 -0400502 if (!sAGpsInterface) {
503 sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE);
Mike Lockwood58bda982009-04-14 16:25:07 -0400504 }
Mike Lockwoode3635c92009-05-11 08:38:02 -0400505 if (sAGpsInterface) {
506 sAGpsInterface->data_conn_closed();
Mike Lockwood58bda982009-04-14 16:25:07 -0400507 }
508}
509
Mike Lockwoode3635c92009-05-11 08:38:02 -0400510static void android_location_GpsLocationProvider_agps_data_conn_failed(JNIEnv* env, jobject obj)
Mike Lockwood58bda982009-04-14 16:25:07 -0400511{
Mike Lockwoode3635c92009-05-11 08:38:02 -0400512 if (!sAGpsInterface) {
513 sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE);
Mike Lockwood58bda982009-04-14 16:25:07 -0400514 }
Mike Lockwoode3635c92009-05-11 08:38:02 -0400515 if (sAGpsInterface) {
516 sAGpsInterface->data_conn_failed();
Mike Lockwood58bda982009-04-14 16:25:07 -0400517 }
518}
519
Mike Lockwoode3635c92009-05-11 08:38:02 -0400520static void android_location_GpsLocationProvider_set_agps_server(JNIEnv* env, jobject obj,
Mike Lockwooda9e54612009-06-19 14:54:42 -0400521 jint type, jstring hostname, jint port)
Mike Lockwood58bda982009-04-14 16:25:07 -0400522{
Mike Lockwoode3635c92009-05-11 08:38:02 -0400523 if (!sAGpsInterface) {
524 sAGpsInterface = (const AGpsInterface*)sGpsInterface->get_extension(AGPS_INTERFACE);
Mike Lockwood58bda982009-04-14 16:25:07 -0400525 }
Mike Lockwoode3635c92009-05-11 08:38:02 -0400526 if (sAGpsInterface) {
Mike Lockwooda9e54612009-06-19 14:54:42 -0400527 const char *c_hostname = env->GetStringUTFChars(hostname, NULL);
528 sAGpsInterface->set_server(type, c_hostname, port);
529 env->ReleaseStringUTFChars(hostname, c_hostname);
Mike Lockwood58bda982009-04-14 16:25:07 -0400530 }
531}
532
Danke Xie22d1f9f2009-08-18 18:28:45 -0400533static void android_location_GpsLocationProvider_send_ni_response(JNIEnv* env, jobject obj,
534 jint notifId, jint response)
535{
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -0600536 if (!sGpsNiInterface) {
537 sGpsNiInterface = (const GpsNiInterface*)sGpsInterface->get_extension(GPS_NI_INTERFACE);
538 }
539 if (sGpsNiInterface) {
540 sGpsNiInterface->respond(notifId, response);
541 }
542}
543
544static jstring android_location_GpsLocationProvider_get_internal_state(JNIEnv* env, jobject obj)
545{
546 jstring result = NULL;
547 if (sGpsDebugInterface) {
548 const size_t maxLength = 2047;
549 char buffer[maxLength+1];
550 size_t length = sGpsDebugInterface->get_internal_state(buffer, maxLength);
551 if (length > maxLength) length = maxLength;
552 buffer[length] = 0;
553 result = env->NewStringUTF(buffer);
554 }
555 return result;
Danke Xie22d1f9f2009-08-18 18:28:45 -0400556}
557
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800558static JNINativeMethod sMethods[] = {
559 /* name, signature, funcPtr */
560 {"class_init_native", "()V", (void *)android_location_GpsLocationProvider_class_init_native},
Mike Lockwoode3635c92009-05-11 08:38:02 -0400561 {"native_is_supported", "()Z", (void*)android_location_GpsLocationProvider_is_supported},
562 {"native_init", "()Z", (void*)android_location_GpsLocationProvider_init},
563 {"native_disable", "()V", (void*)android_location_GpsLocationProvider_disable},
564 {"native_cleanup", "()V", (void*)android_location_GpsLocationProvider_cleanup},
Mike Lockwood04598b62010-04-14 17:17:24 -0400565 {"native_set_position_mode", "(IIIII)Z", (void*)android_location_GpsLocationProvider_set_position_mode},
566 {"native_start", "()Z", (void*)android_location_GpsLocationProvider_start},
Mike Lockwoode3635c92009-05-11 08:38:02 -0400567 {"native_stop", "()Z", (void*)android_location_GpsLocationProvider_stop},
568 {"native_delete_aiding_data", "(I)V", (void*)android_location_GpsLocationProvider_delete_aiding_data},
569 {"native_wait_for_event", "()V", (void*)android_location_GpsLocationProvider_wait_for_event},
570 {"native_read_sv_status", "([I[F[F[F[I)I", (void*)android_location_GpsLocationProvider_read_sv_status},
Mike Lockwoodb16e7802009-08-06 09:26:02 -0400571 {"native_read_nmea", "(I[BI)I", (void*)android_location_GpsLocationProvider_read_nmea},
Mike Lockwoode3635c92009-05-11 08:38:02 -0400572 {"native_inject_time", "(JJI)V", (void*)android_location_GpsLocationProvider_inject_time},
Mike Lockwoodd26ce0d2009-06-11 12:25:46 -0400573 {"native_inject_location", "(DDF)V", (void*)android_location_GpsLocationProvider_inject_location},
Mike Lockwoode3635c92009-05-11 08:38:02 -0400574 {"native_supports_xtra", "()Z", (void*)android_location_GpsLocationProvider_supports_xtra},
575 {"native_inject_xtra_data", "([BI)V", (void*)android_location_GpsLocationProvider_inject_xtra_data},
576 {"native_agps_data_conn_open", "(Ljava/lang/String;)V", (void*)android_location_GpsLocationProvider_agps_data_conn_open},
577 {"native_agps_data_conn_closed", "()V", (void*)android_location_GpsLocationProvider_agps_data_conn_closed},
578 {"native_agps_data_conn_failed", "()V", (void*)android_location_GpsLocationProvider_agps_data_conn_failed},
Mike Lockwooda9e54612009-06-19 14:54:42 -0400579 {"native_set_agps_server", "(ILjava/lang/String;I)V", (void*)android_location_GpsLocationProvider_set_agps_server},
Danke Xie22d1f9f2009-08-18 18:28:45 -0400580 {"native_send_ni_response", "(II)V", (void*)android_location_GpsLocationProvider_send_ni_response},
Fred Fettinger3c8fbdf2010-01-04 15:38:13 -0600581 {"native_get_internal_state", "()Ljava/lang/String;", (void*)android_location_GpsLocationProvider_get_internal_state},
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800582};
583
Mike Lockwood00b74272010-03-26 10:41:48 -0400584int register_android_server_location_GpsLocationProvider(JNIEnv* env)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800585{
Mike Lockwood00b74272010-03-26 10:41:48 -0400586 return jniRegisterNativeMethods(env, "com/android/server/location/GpsLocationProvider", sMethods, NELEM(sMethods));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800587}
588
589} /* namespace android */