• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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, &notification, &notificationLen)) {
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, &notification, &notificationLen)) {
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