1 /*
2 * Copyright (C) 2020 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 #include "chpp/services/gnss.h"
18
19 #include <inttypes.h>
20 #include <stddef.h>
21 #include <stdint.h>
22
23 #include "chpp/common/gnss.h"
24 #include "chpp/common/gnss_types.h"
25 #include "chpp/common/standard_uuids.h"
26 #include "chpp/log.h"
27 #include "chpp/macros.h"
28 #include "chpp/services.h"
29 #include "chre/pal/gnss.h"
30
31 /************************************************
32 * Prototypes
33 ***********************************************/
34
35 static enum ChppAppErrorCode chppDispatchGnssRequest(void *serviceContext,
36 uint8_t *buf, size_t len);
37 static void chppGnssServiceNotifyReset(void *serviceContext);
38
39 /************************************************
40 * Private Definitions
41 ***********************************************/
42
43 /**
44 * Configuration parameters for this service
45 */
46 static const struct ChppService kGnssServiceConfig = {
47 .descriptor.uuid = CHPP_UUID_GNSS_STANDARD,
48
49 // Human-readable name
50 .descriptor.name = "GNSS",
51
52 // Version
53 .descriptor.version.major = 1,
54 .descriptor.version.minor = 0,
55 .descriptor.version.patch = 0,
56
57 // Notifies service if CHPP is reset
58 .resetNotifierFunctionPtr = &chppGnssServiceNotifyReset,
59
60 // Client request dispatch function pointer
61 .requestDispatchFunctionPtr = &chppDispatchGnssRequest,
62
63 // Client notification dispatch function pointer
64 .notificationDispatchFunctionPtr = NULL, // Not supported
65
66 // Min length is the entire header
67 .minLength = sizeof(struct ChppAppHeader),
68 };
69
70 /**
71 * Structure to maintain state for the GNSS service and its Request/Response
72 * (RR) functionality.
73 */
74 struct ChppGnssServiceState {
75 struct ChppServiceState service; // GNSS service state
76 const struct chrePalGnssApi *api; // GNSS PAL API
77
78 // Based on chre/pal/gnss.h and chrePalGnssApi
79 struct ChppRequestResponseState open; // Service init state
80 struct ChppRequestResponseState close; // Service deinit state
81 struct ChppRequestResponseState getCapabilities; // Get Capabilities state
82 struct ChppRequestResponseState
83 controlLocationSession; // Control Location measurement state
84 struct ChppRequestResponseState
85 controlMeasurementSession; // Control Raw GNSS measurement state
86 struct ChppRequestResponseState
87 configurePassiveLocationListener; // Configure Passive location receiving
88 // state
89 };
90
91 // Note: The CHRE PAL API only allows for one definition - see comment in WWAN
92 // service for details.
93 // Note: There is no notion of a cookie in the CHRE GNSS API so we need to use
94 // the global service state (gGnssServiceContext) directly in all callbacks.
95 struct ChppGnssServiceState gGnssServiceContext;
96
97 /************************************************
98 * Prototypes
99 ***********************************************/
100
101 static enum ChppAppErrorCode chppGnssServiceOpen(
102 struct ChppGnssServiceState *gnssServiceContext,
103 struct ChppAppHeader *requestHeader);
104 static enum ChppAppErrorCode chppGnssServiceClose(
105 struct ChppGnssServiceState *gnssServiceContext,
106 struct ChppAppHeader *requestHeader);
107 static enum ChppAppErrorCode chppGnssServiceGetCapabilities(
108 struct ChppGnssServiceState *gnssServiceContext,
109 struct ChppAppHeader *requestHeader);
110 static enum ChppAppErrorCode chppGnssServiceControlLocationSession(
111 struct ChppGnssServiceState *gnssServiceContext,
112 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len);
113 static enum ChppAppErrorCode chppGnssServiceControlMeasurementSession(
114 struct ChppGnssServiceState *gnssServiceContext,
115 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len);
116 static enum ChppAppErrorCode chppGnssServiceConfigurePassiveLocationListener(
117 struct ChppGnssServiceState *gnssServiceContext,
118 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len);
119
120 static void chppGnssServiceRequestStateResyncCallback(void);
121 static void chppGnssServiceLocationStatusChangeCallback(bool enabled,
122 uint8_t errorCode);
123 static void chppGnssServiceLocationEventCallback(
124 struct chreGnssLocationEvent *event);
125 static void chppGnssServiceMeasurementStatusChangeCallback(bool enabled,
126 uint8_t errorCode);
127 static void chppGnssServiceMeasurementEventCallback(
128 struct chreGnssDataEvent *event);
129
130 /************************************************
131 * Private Functions
132 ***********************************************/
133
134 /**
135 * Dispatches a client request from the transport layer that is determined to be
136 * for the GNSS service. If the result of the dispatch is an error, this
137 * function responds to the client with the same error.
138 *
139 * This function is called from the app layer using its function pointer given
140 * during service registration.
141 *
142 * @param serviceContext Maintains status for each service instance.
143 * @param buf Input data. Cannot be null.
144 * @param len Length of input data in bytes.
145 *
146 * @return Indicates the result of this function call.
147 */
chppDispatchGnssRequest(void * serviceContext,uint8_t * buf,size_t len)148 static enum ChppAppErrorCode chppDispatchGnssRequest(void *serviceContext,
149 uint8_t *buf, size_t len) {
150 struct ChppAppHeader *rxHeader = (struct ChppAppHeader *)buf;
151 buf += sizeof(struct ChppAppHeader);
152 len -= sizeof(struct ChppAppHeader);
153
154 struct ChppGnssServiceState *gnssServiceContext =
155 (struct ChppGnssServiceState *)serviceContext;
156 struct ChppRequestResponseState *rRState = NULL;
157 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
158 bool dispatched = true;
159
160 switch (rxHeader->command) {
161 case CHPP_GNSS_OPEN: {
162 rRState = &gnssServiceContext->open;
163 chppServiceTimestampRequest(rRState, rxHeader);
164 error = chppGnssServiceOpen(gnssServiceContext, rxHeader);
165 break;
166 }
167
168 case CHPP_GNSS_CLOSE: {
169 rRState = &gnssServiceContext->close;
170 chppServiceTimestampRequest(rRState, rxHeader);
171 error = chppGnssServiceClose(gnssServiceContext, rxHeader);
172 break;
173 }
174
175 case CHPP_GNSS_GET_CAPABILITIES: {
176 rRState = &gnssServiceContext->getCapabilities;
177 chppServiceTimestampRequest(rRState, rxHeader);
178 error = chppGnssServiceGetCapabilities(gnssServiceContext, rxHeader);
179 break;
180 }
181
182 case CHPP_GNSS_CONTROL_LOCATION_SESSION: {
183 rRState = &gnssServiceContext->controlLocationSession;
184 chppServiceTimestampRequest(rRState, rxHeader);
185 error = chppGnssServiceControlLocationSession(gnssServiceContext,
186 rxHeader, buf, len);
187 break;
188 }
189
190 case CHPP_GNSS_CONTROL_MEASUREMENT_SESSION: {
191 rRState = &gnssServiceContext->controlMeasurementSession;
192 chppServiceTimestampRequest(rRState, rxHeader);
193 error = chppGnssServiceControlMeasurementSession(gnssServiceContext,
194 rxHeader, buf, len);
195 break;
196 }
197
198 case CHPP_GNSS_CONFIGURE_PASSIVE_LOCATION_LISTENER: {
199 rRState = &gnssServiceContext->configurePassiveLocationListener;
200 chppServiceTimestampRequest(rRState, rxHeader);
201 error = chppGnssServiceConfigurePassiveLocationListener(
202 gnssServiceContext, rxHeader, buf, len);
203 break;
204 }
205
206 default: {
207 dispatched = false;
208 error = CHPP_APP_ERROR_INVALID_COMMAND;
209 break;
210 }
211 }
212
213 if (dispatched == true && error != CHPP_APP_ERROR_NONE) {
214 // Request was dispatched but an error was returned. Close out
215 // chppServiceTimestampRequest()
216 chppServiceTimestampResponse(rRState);
217 }
218
219 return error;
220 }
221
222 /**
223 * Initializes the GNSS service upon an open request from the client and
224 * responds to the client with the result.
225 *
226 * @param serviceContext Maintains status for each service instance.
227 * @param requestHeader App layer header of the request.
228 *
229 * @return Indicates the result of this function call.
230 */
chppGnssServiceOpen(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader)231 static enum ChppAppErrorCode chppGnssServiceOpen(
232 struct ChppGnssServiceState *gnssServiceContext,
233 struct ChppAppHeader *requestHeader) {
234 static const struct chrePalGnssCallbacks palCallbacks = {
235 .requestStateResync = chppGnssServiceRequestStateResyncCallback,
236 .locationStatusChangeCallback =
237 chppGnssServiceLocationStatusChangeCallback,
238 .locationEventCallback = chppGnssServiceLocationEventCallback,
239 .measurementStatusChangeCallback =
240 chppGnssServiceMeasurementStatusChangeCallback,
241 .measurementEventCallback = chppGnssServiceMeasurementEventCallback,
242 };
243
244 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
245
246 if (gnssServiceContext->service.openState == CHPP_OPEN_STATE_OPENED) {
247 CHPP_DEBUG_ASSERT_LOG(false, "GNSS service already open");
248 error = CHPP_APP_ERROR_INVALID_COMMAND;
249
250 } else if (!gnssServiceContext->api->open(
251 gnssServiceContext->service.appContext->systemApi,
252 &palCallbacks)) {
253 CHPP_DEBUG_ASSERT_LOG(false, "GNSS PAL open failed");
254 error = CHPP_APP_ERROR_BEYOND_CHPP;
255
256 } else {
257 CHPP_LOGD("GNSS service opened");
258 gnssServiceContext->service.openState = CHPP_OPEN_STATE_OPENED;
259
260 struct ChppAppHeader *response =
261 chppAllocServiceResponseFixed(requestHeader, struct ChppAppHeader);
262 size_t responseLen = sizeof(*response);
263
264 if (response == NULL) {
265 CHPP_LOG_OOM();
266 error = CHPP_APP_ERROR_OOM;
267 } else {
268 chppSendTimestampedResponseOrFail(&gnssServiceContext->service,
269 &gnssServiceContext->open, response,
270 responseLen);
271 }
272 }
273
274 return error;
275 }
276
277 /**
278 * Deinitializes the GNSS service.
279 *
280 * @param serviceContext Maintains status for each service instance.
281 * @param requestHeader App layer header of the request.
282 *
283 * @return Indicates the result of this function call.
284 */
chppGnssServiceClose(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader)285 static enum ChppAppErrorCode chppGnssServiceClose(
286 struct ChppGnssServiceState *gnssServiceContext,
287 struct ChppAppHeader *requestHeader) {
288 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
289
290 gnssServiceContext->api->close();
291 gnssServiceContext->service.openState = CHPP_OPEN_STATE_CLOSED;
292
293 CHPP_LOGD("GNSS service closed");
294
295 struct ChppAppHeader *response =
296 chppAllocServiceResponseFixed(requestHeader, struct ChppAppHeader);
297 size_t responseLen = sizeof(*response);
298
299 if (response == NULL) {
300 CHPP_LOG_OOM();
301 error = CHPP_APP_ERROR_OOM;
302 } else {
303 chppSendTimestampedResponseOrFail(&gnssServiceContext->service,
304 &gnssServiceContext->close, response,
305 responseLen);
306 }
307
308 return error;
309 }
310
311 /**
312 * Notifies the service of an incoming reset.
313 *
314 * @param serviceContext Maintains status for each service instance.
315 */
chppGnssServiceNotifyReset(void * serviceContext)316 static void chppGnssServiceNotifyReset(void *serviceContext) {
317 struct ChppGnssServiceState *gnssServiceContext =
318 (struct ChppGnssServiceState *)serviceContext;
319
320 if (gnssServiceContext->service.openState != CHPP_OPEN_STATE_OPENED) {
321 CHPP_LOGW("GNSS service reset but wasn't open");
322 } else {
323 CHPP_LOGD("GNSS service reset. Closing");
324 gnssServiceContext->service.openState = CHPP_OPEN_STATE_CLOSED;
325 gnssServiceContext->api->close();
326 }
327 }
328
329 /**
330 * Retrieves a set of flags indicating the GNSS features supported by the
331 * current implementation.
332 *
333 * @param serviceContext Maintains status for each service instance.
334 * @param requestHeader App layer header of the request.
335 *
336 * @return Indicates the result of this function call.
337 */
chppGnssServiceGetCapabilities(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader)338 static enum ChppAppErrorCode chppGnssServiceGetCapabilities(
339 struct ChppGnssServiceState *gnssServiceContext,
340 struct ChppAppHeader *requestHeader) {
341 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
342
343 struct ChppGnssGetCapabilitiesResponse *response =
344 chppAllocServiceResponseFixed(requestHeader,
345 struct ChppGnssGetCapabilitiesResponse);
346 size_t responseLen = sizeof(*response);
347
348 if (response == NULL) {
349 CHPP_LOG_OOM();
350 error = CHPP_APP_ERROR_OOM;
351 } else {
352 response->params.capabilities = gnssServiceContext->api->getCapabilities();
353
354 CHPP_LOGD("chppGnssServiceGetCapabilities returning 0x%" PRIx32
355 ", %" PRIuSIZE " bytes",
356 response->params.capabilities, responseLen);
357 chppSendTimestampedResponseOrFail(&gnssServiceContext->service,
358 &gnssServiceContext->getCapabilities,
359 response, responseLen);
360 }
361
362 return error;
363 }
364
365 /**
366 * Start/stop/modify the GNSS location session.
367 *
368 * This function returns an error code synchronously.
369 * A subsequent call to chppGnssServiceLocationStatusChangeCallback() will be
370 * used to communicate the result of this request (as a service response).
371 * A subsequent call to chppGnssServiceLocationEventCallback() will be used to
372 * communicate the location fixes (as service notifications).
373 *
374 * @param serviceContext Maintains status for each service instance.
375 * @param requestHeader App layer header of the request.
376 * @param buf Input data. Cannot be null.
377 * @param len Length of input data in bytes.
378 *
379 * @return Indicates the result of this function call.
380 */
chppGnssServiceControlLocationSession(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader,uint8_t * buf,size_t len)381 static enum ChppAppErrorCode chppGnssServiceControlLocationSession(
382 struct ChppGnssServiceState *gnssServiceContext,
383 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len) {
384 UNUSED_VAR(requestHeader);
385 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
386
387 if (len < sizeof(struct ChppGnssControlLocationSessionParameters)) {
388 error = CHPP_APP_ERROR_INVALID_ARG;
389
390 } else {
391 struct ChppGnssControlLocationSessionParameters *parameters =
392 (struct ChppGnssControlLocationSessionParameters *)buf;
393
394 if (!gnssServiceContext->api->controlLocationSession(
395 parameters->enable, parameters->minIntervalMs,
396 parameters->minTimeToNextFixMs)) {
397 error = CHPP_APP_ERROR_UNSPECIFIED;
398 }
399 }
400
401 return error;
402 }
403
404 /**
405 * Start/stop/modify the raw GNSS measurement session.
406 *
407 * This function returns an error code synchronously.
408 * A subsequent call to chppGnssServiceMeasurementStatusChangeCallback() will be
409 * used to communicate the result of this request (as a service response).
410 * A subsequent call to chppGnssServiceMeasurementEventCallback() will be used
411 * to communicate the measurements (as service notifications).
412 *
413 * @param serviceContext Maintains status for each service instance.
414 * @param requestHeader App layer header of the request.
415 * @param buf Input data. Cannot be null.
416 * @param len Length of input data in bytes.
417 *
418 * @return Indicates the result of this function call.
419 */
chppGnssServiceControlMeasurementSession(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader,uint8_t * buf,size_t len)420 static enum ChppAppErrorCode chppGnssServiceControlMeasurementSession(
421 struct ChppGnssServiceState *gnssServiceContext,
422 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len) {
423 UNUSED_VAR(requestHeader);
424 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
425
426 if (len < sizeof(struct ChppGnssControlMeasurementSessionParameters)) {
427 error = CHPP_APP_ERROR_INVALID_ARG;
428
429 } else {
430 struct ChppGnssControlMeasurementSessionParameters *parameters =
431 (struct ChppGnssControlMeasurementSessionParameters *)buf;
432
433 if (!gnssServiceContext->api->controlMeasurementSession(
434 parameters->enable, parameters->minIntervalMs)) {
435 error = CHPP_APP_ERROR_UNSPECIFIED;
436 }
437 }
438
439 return error;
440 }
441
442 /**
443 * Configures whether to opportunistically deliver any location fixes produced
444 * for other clients of the GNSS engine.
445 *
446 * This function returns an error code synchronously.
447 * A subsequent call to chppGnssServiceLocationEventCallback() will be used to
448 * communicate the location fixes (as service notifications).
449 *
450 * @param serviceContext Maintains status for each service instance.
451 * @param requestHeader App layer header of the request.
452 * @param buf Input data. Cannot be null.
453 * @param len Length of input data in bytes.
454 *
455 * @return Indicates the result of this function call.
456 */
chppGnssServiceConfigurePassiveLocationListener(struct ChppGnssServiceState * gnssServiceContext,struct ChppAppHeader * requestHeader,uint8_t * buf,size_t len)457 static enum ChppAppErrorCode chppGnssServiceConfigurePassiveLocationListener(
458 struct ChppGnssServiceState *gnssServiceContext,
459 struct ChppAppHeader *requestHeader, uint8_t *buf, size_t len) {
460 UNUSED_VAR(requestHeader);
461 enum ChppAppErrorCode error = CHPP_APP_ERROR_NONE;
462
463 if (len < sizeof(struct ChppGnssConfigurePassiveLocationListenerParameters)) {
464 error = CHPP_APP_ERROR_INVALID_ARG;
465 } else {
466 struct ChppGnssConfigurePassiveLocationListenerParameters *parameters =
467 (struct ChppGnssConfigurePassiveLocationListenerParameters *)buf;
468
469 if (!gnssServiceContext->api->configurePassiveLocationListener(
470 parameters->enable)) {
471 error = CHPP_APP_ERROR_UNSPECIFIED;
472
473 } else {
474 struct ChppAppHeader *response =
475 chppAllocServiceResponseFixed(requestHeader, struct ChppAppHeader);
476 size_t responseLen = sizeof(*response);
477
478 if (response == NULL) {
479 CHPP_LOG_OOM();
480 error = CHPP_APP_ERROR_OOM;
481 } else {
482 chppSendTimestampedResponseOrFail(
483 &gnssServiceContext->service,
484 &gnssServiceContext->configurePassiveLocationListener, response,
485 responseLen);
486 }
487 }
488 }
489
490 return error;
491 }
492
493 /**
494 * GNSS PAL callback to request that the core CHRE system re-send requests for
495 * any active sessions and its current passive location listener setting.
496 */
chppGnssServiceRequestStateResyncCallback(void)497 static void chppGnssServiceRequestStateResyncCallback(void) {
498 struct ChppAppHeader *notification =
499 chppAllocServiceNotificationFixed(struct ChppAppHeader);
500 size_t notificationLen = sizeof(*notification);
501
502 if (notification == NULL) {
503 CHPP_LOG_OOM();
504 CHPP_ASSERT(false);
505
506 } else {
507 notification->handle = gGnssServiceContext.service.handle;
508 notification->command = CHPP_GNSS_REQUEST_STATE_RESYNC_NOTIFICATION;
509
510 chppEnqueueTxDatagramOrFail(
511 gGnssServiceContext.service.appContext->transportContext, notification,
512 notificationLen);
513 }
514 }
515
516 /**
517 * GNSS PAL callback to inform the CHRE of the result of changes to the location
518 * session status.
519 */
chppGnssServiceLocationStatusChangeCallback(bool enabled,uint8_t errorCode)520 static void chppGnssServiceLocationStatusChangeCallback(bool enabled,
521 uint8_t errorCode) {
522 // Recreate request header
523 struct ChppAppHeader requestHeader = {
524 .handle = gGnssServiceContext.service.handle,
525 .transaction = gGnssServiceContext.controlLocationSession.transaction,
526 .command = CHPP_GNSS_CONTROL_LOCATION_SESSION,
527 };
528
529 struct ChppGnssControlLocationSessionResponse *response =
530 chppAllocServiceResponseFixed(
531 &requestHeader, struct ChppGnssControlLocationSessionResponse);
532 size_t responseLen = sizeof(*response);
533
534 if (response == NULL) {
535 CHPP_LOG_OOM();
536 CHPP_ASSERT(false);
537
538 } else {
539 response->enabled = enabled;
540 response->errorCode = errorCode;
541
542 chppSendTimestampedResponseOrFail(
543 &gGnssServiceContext.service,
544 &gGnssServiceContext.controlLocationSession, response, responseLen);
545 }
546 }
547
548 /**
549 * GNSS PAL callback to pass GNSS location fixes to the core CHRE system.
550 */
chppGnssServiceLocationEventCallback(struct chreGnssLocationEvent * event)551 static void chppGnssServiceLocationEventCallback(
552 struct chreGnssLocationEvent *event) {
553 // Craft response per parser script
554 struct ChppGnssLocationEventWithHeader *notification = NULL;
555 size_t notificationLen = 0;
556
557 if (!chppGnssLocationEventFromChre(event, ¬ification, ¬ificationLen)) {
558 CHPP_LOGE("LocationEvent conversion failed (OOM?)");
559
560 notification = chppMalloc(sizeof(struct ChppAppHeader));
561 if (notification == NULL) {
562 CHPP_LOG_OOM();
563 } else {
564 notificationLen = sizeof(struct ChppAppHeader);
565 }
566 }
567
568 if (notification != NULL) {
569 notification->header.handle = gGnssServiceContext.service.handle;
570 notification->header.type = CHPP_MESSAGE_TYPE_SERVICE_NOTIFICATION;
571 notification->header.transaction =
572 0; // Because we don't know this is in response to a Location Session
573 // or Passive Location Listener
574 notification->header.error =
575 (notificationLen > sizeof(struct ChppAppHeader))
576 ? CHPP_APP_ERROR_NONE
577 : CHPP_APP_ERROR_CONVERSION_FAILED;
578 notification->header.command = CHPP_GNSS_LOCATION_RESULT_NOTIFICATION;
579
580 chppEnqueueTxDatagramOrFail(
581 gGnssServiceContext.service.appContext->transportContext, notification,
582 notificationLen);
583 }
584
585 gGnssServiceContext.api->releaseLocationEvent(event);
586 }
587
588 /**
589 * GNSS PAL callback to inform the CHRE of the result of changes to the raw GNSS
590 * measurement session status.
591 */
chppGnssServiceMeasurementStatusChangeCallback(bool enabled,uint8_t errorCode)592 static void chppGnssServiceMeasurementStatusChangeCallback(bool enabled,
593 uint8_t errorCode) {
594 // Recreate request header
595 struct ChppAppHeader requestHeader = {
596 .handle = gGnssServiceContext.service.handle,
597 .transaction = gGnssServiceContext.controlMeasurementSession.transaction,
598 .command = CHPP_GNSS_CONTROL_MEASUREMENT_SESSION,
599 };
600
601 struct ChppGnssControlMeasurementSessionResponse *response =
602 chppAllocServiceResponseFixed(
603 &requestHeader, struct ChppGnssControlMeasurementSessionResponse);
604 size_t responseLen = sizeof(*response);
605
606 if (response == NULL) {
607 CHPP_LOG_OOM();
608 CHPP_ASSERT(false);
609
610 } else {
611 response->enabled = enabled;
612 response->errorCode = errorCode;
613
614 chppSendTimestampedResponseOrFail(
615 &gGnssServiceContext.service,
616 &gGnssServiceContext.controlMeasurementSession, response, responseLen);
617 }
618 }
619
620 /**
621 * GNSS PAL callback to pass raw GNSS measurement data to the core CHRE system.
622 */
chppGnssServiceMeasurementEventCallback(struct chreGnssDataEvent * event)623 static void chppGnssServiceMeasurementEventCallback(
624 struct chreGnssDataEvent *event) {
625 // Craft response per parser script
626 struct ChppGnssDataEventWithHeader *notification = NULL;
627 size_t notificationLen = 0;
628
629 if (!chppGnssDataEventFromChre(event, ¬ification, ¬ificationLen)) {
630 CHPP_LOGE("DataEvent conversion failed (OOM?) ID=%" PRIu8,
631 gGnssServiceContext.controlMeasurementSession.transaction);
632
633 notification = chppMalloc(sizeof(struct ChppAppHeader));
634 if (notification == NULL) {
635 CHPP_LOG_OOM();
636 } else {
637 notificationLen = sizeof(struct ChppAppHeader);
638 }
639 }
640
641 if (notification != NULL) {
642 notification->header.handle = gGnssServiceContext.service.handle;
643 notification->header.type = CHPP_MESSAGE_TYPE_SERVICE_NOTIFICATION;
644 notification->header.transaction =
645 gGnssServiceContext.controlMeasurementSession.transaction;
646 notification->header.error =
647 (notificationLen > sizeof(struct ChppAppHeader))
648 ? CHPP_APP_ERROR_NONE
649 : CHPP_APP_ERROR_CONVERSION_FAILED;
650 notification->header.command = CHPP_GNSS_MEASUREMENT_RESULT_NOTIFICATION;
651
652 chppEnqueueTxDatagramOrFail(
653 gGnssServiceContext.service.appContext->transportContext, notification,
654 notificationLen);
655 }
656
657 gGnssServiceContext.api->releaseMeasurementDataEvent(event);
658 }
659
660 /************************************************
661 * Public Functions
662 ***********************************************/
663
chppRegisterGnssService(struct ChppAppState * appContext)664 void chppRegisterGnssService(struct ChppAppState *appContext) {
665 gGnssServiceContext.api = chrePalGnssGetApi(CHPP_PAL_GNSS_API_VERSION);
666
667 if (gGnssServiceContext.api == NULL) {
668 CHPP_DEBUG_ASSERT_LOG(false,
669 "GNSS PAL API incompatible. Cannot register service");
670
671 } else {
672 chppRegisterService(appContext, (void *)&gGnssServiceContext,
673 &gGnssServiceContext.service, &kGnssServiceConfig);
674 CHPP_DEBUG_ASSERT(gGnssServiceContext.service.handle);
675 }
676 }
677
chppDeregisterGnssService(struct ChppAppState * appContext)678 void chppDeregisterGnssService(struct ChppAppState *appContext) {
679 // TODO
680
681 UNUSED_VAR(appContext);
682 }
683