blob: dee3fdd97f691be049f202c99e10a956e8c4f877 [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
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
29static pthread_mutex_t sEventMutex = PTHREAD_MUTEX_INITIALIZER;
30static pthread_cond_t sEventCond = PTHREAD_COND_INITIALIZER;
31static jmethodID method_reportLocation;
32static jmethodID method_reportStatus;
33static jmethodID method_reportSvStatus;
34static jmethodID method_xtraDownloadRequest;
35
36static const GpsInterface* sGpsInterface = NULL;
37static const GpsXtraInterface* sGpsXtraInterface = NULL;
38
39// data written to by GPS callbacks
40static GpsLocation sGpsLocation;
41static GpsStatus sGpsStatus;
42static GpsSvStatus sGpsSvStatus;
43
44// a copy of the data shared by android_location_GpsLocationProvider_wait_for_event
45// and android_location_GpsLocationProvider_read_status
46static GpsLocation sGpsLocationCopy;
47static GpsStatus sGpsStatusCopy;
48static GpsSvStatus sGpsSvStatusCopy;
49
50enum CallbackType {
51 kLocation = 1,
52 kStatus = 2,
53 kSvStatus = 4,
54 kXtraDownloadRequest = 8,
55 kDisableRequest = 16,
56};
57static int sPendingCallbacks;
58
59namespace android {
60
61static 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
72static 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
83static 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
94GpsCallbacks sGpsCallbacks = {
95 location_callback,
96 status_callback,
97 sv_status_callback,
98};
99
100static void
101download_request_callback()
102{
103 pthread_mutex_lock(&sEventMutex);
104 sPendingCallbacks |= kXtraDownloadRequest;
105 pthread_cond_signal(&sEventCond);
106 pthread_mutex_unlock(&sEventMutex);
107}
108
109GpsXtraCallbacks sGpsXtraCallbacks = {
110 download_request_callback,
111};
112
113
114static 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
121static jboolean android_location_GpsLocationProvider_is_supported(JNIEnv* env, jclass clazz) {
122 if (!sGpsInterface)
123 sGpsInterface = gps_get_interface();
124 return (sGpsInterface != NULL);
125}
126
127static 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
134static 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
142static void android_location_GpsLocationProvider_cleanup(JNIEnv* env, jobject obj)
143{
144 sGpsInterface->cleanup();
145}
146
147static 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
157static jboolean android_location_GpsLocationProvider_stop(JNIEnv* env, jobject obj)
158{
159 return (sGpsInterface->stop() == 0);
160}
161
162static 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
168static void android_location_GpsLocationProvider_delete_aiding_data(JNIEnv* env, jobject obj, jint flags)
169{
170 sGpsInterface->delete_aiding_data(flags);
171}
172
173static 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
210static 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
241static 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
247static 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
262static 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
270static 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
288int register_android_location_GpsLocationProvider(JNIEnv* env)
289{
290 return jniRegisterNativeMethods(env, "com/android/internal/location/GpsLocationProvider", sMethods, NELEM(sMethods));
291}
292
293} /* namespace android */