1 /* 2 * Copyright (C) 2018 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 #ifndef CHRE_CORE_GNSS_MANAGER_H_ 18 #define CHRE_CORE_GNSS_MANAGER_H_ 19 20 #include <cstdint> 21 22 #include "chre/core/api_manager_common.h" 23 #include "chre/core/nanoapp.h" 24 #include "chre/core/settings.h" 25 #include "chre/platform/platform_gnss.h" 26 #include "chre/util/non_copyable.h" 27 #include "chre/util/system/debug_dump.h" 28 #include "chre/util/time.h" 29 30 namespace chre { 31 32 class GnssManager; 33 34 /** 35 * A helper class that manages requests for a GNSS location or measurement 36 * session. 37 */ 38 class GnssSession { 39 public: 40 /** 41 * Adds a request to a session asynchronously. The result is delivered 42 * through a CHRE_EVENT_GNSS_ASYNC_RESULT event. 43 * 44 * @param nanoapp The nanoapp adding the request. 45 * @param minInterval The minimum reporting interval for results. 46 * @param timeToNext The amount of time that the GNSS system is allowed to 47 * delay generating a report. 48 * @param cookie A cookie that is round-tripped to provide context to the 49 * nanoapp making the request. 50 * 51 * @return true if the request was accepted for processing. 52 */ 53 bool addRequest(Nanoapp *nanoapp, Milliseconds minInterval, 54 Milliseconds minTimeToNext, const void *cookie); 55 56 /** 57 * Removes a request from a session asynchronously. The result is delivered 58 * through a CHRE_EVENT_GNSS_ASYNC_RESULT event. 59 * 60 * @param nanoapp The nanoapp removing the request. 61 * @param cookie A cookie that is round-tripped to provide context to the 62 * nanoapp making the request. 63 * 64 * @return true if the request was accepted for processing. 65 */ 66 bool removeRequest(Nanoapp *nanoapp, const void *cookie); 67 68 /** 69 * Checks if a nanoapp has an open session request. 70 * 71 * @param nanoapp The nanoapp removing the request. 72 * 73 * @return whether the nanoapp has an active request. 74 */ 75 bool nanoappHasRequest(Nanoapp *nanoapp) const; 76 77 /** 78 * Handles the result of a request to the PlatformGnss to request a change to 79 * the session. 80 * 81 * @param enabled true if the session is currently active. 82 * @param errorCode an error code that is used to indicate success or what 83 * type of error has occured. See chreError enum in the CHRE API for 84 * additional details. 85 */ 86 void handleStatusChange(bool enabled, uint8_t errorCode); 87 88 /** 89 * Handles a CHRE GNSS report (location/data) event. 90 * 91 * @param event The GNSS report event provided to the GNSS session. This 92 * memory is guaranteed not to be modified until it has been explicitly 93 * released through the PlatformGnss instance. 94 */ 95 void handleReportEvent(void *event); 96 97 /** 98 * @return true if an async response is pending from GNSS. This method should 99 * be used to check if a GNSS session request is in flight. 100 */ asyncResponsePending()101 bool asyncResponsePending() const { 102 return !mStateTransitions.empty() || mInternalRequestPending; 103 } 104 105 /** 106 * Invoked when the host notifies CHRE of a settings change. 107 * 108 * @param setting The setting that changed. 109 * @param enabled Whether setting is enabled or not. 110 */ 111 void onSettingChanged(Setting setting, bool enabled); 112 113 /** 114 * Updates the platform GNSS request according to the current state. It should 115 * be used to synchronize the GNSS to the desired state, e.g. for setting 116 * updates or handling a state resync request. 117 * 118 * @param forceUpdate If true, force the platform GNSS request to be made. 119 * 120 * @return true if the invocation resulted in dispatching an internal 121 * request to control the platform layer 122 */ 123 bool updatePlatformRequest(bool forceUpdate = false); 124 125 /** 126 * Invoked as a result of a requestStateResync() callback from the GNSS PAL. 127 * Runs in the context of the CHRE thread. 128 */ 129 void handleRequestStateResyncCallbackSync(); 130 131 /** 132 * Prints state in a string buffer. Must only be called from the context of 133 * the main CHRE thread. 134 * 135 * @param debugDump The debug dump wrapper where a string can be printed 136 * into one of the buffers. 137 */ 138 void logStateToBuffer(DebugDumpWrapper &debugDump) const; 139 140 private: 141 /** 142 * Tracks a nanoapp that has subscribed to a session and the reporting 143 * interval. 144 */ 145 struct Request { 146 //! The nanoapp instance ID that made this request. 147 uint32_t nanoappInstanceId; 148 149 //! The interval of results requested. 150 Milliseconds minInterval; 151 }; 152 153 //! Internal struct with data needed to log last X session requests 154 struct SessionRequestLog { SessionRequestLogSessionRequestLog155 SessionRequestLog(Nanoseconds timestampIn, uint16_t instanceIdIn, 156 Milliseconds intervalIn, bool startIn) 157 : timestamp(timestampIn), 158 instanceId(instanceIdIn), 159 interval(intervalIn), 160 start(startIn) {} 161 Nanoseconds timestamp; 162 uint16_t instanceId; 163 Milliseconds interval; 164 bool start; 165 }; 166 167 /** 168 * Tracks the state of the GNSS engine. 169 */ 170 struct StateTransition { 171 //! The cookie provided to the CHRE API when the nanoapp requested a 172 //! change to the state of the GNSS engine. 173 const void *cookie; 174 175 //! The nanoapp instance ID that prompted the change. 176 uint16_t nanoappInstanceId; 177 178 //! The target state of the GNSS engine. 179 bool enable; 180 181 //! The target minimum reporting interval for the GNSS engine. This is only 182 //! valid if enable is set to true. 183 Milliseconds minInterval; 184 }; 185 186 //! The event type of the session's report data. 187 const uint16_t kReportEventType; 188 189 //! The request type to start and stop a session. 190 uint8_t mStartRequestType; 191 uint8_t mStopRequestType; 192 193 //! The session name, used in logging state. 194 const char *mName; 195 196 //! The maximum number of pending state transitions allowed. 197 static constexpr size_t kMaxGnssStateTransitions = 8; 198 199 //! The queue of state transitions for the GNSS engine. Only one asynchronous 200 //! state transition can be in flight at one time. Any further requests are 201 //! queued here. 202 ArrayQueue<StateTransition, kMaxGnssStateTransitions> mStateTransitions; 203 204 //! The list of most recent session request logs 205 static constexpr size_t kNumSessionRequestLogs = 10; 206 ArrayQueue<SessionRequestLog, kNumSessionRequestLogs> mSessionRequestLogs; 207 208 //! The request multiplexer for GNSS session requests. 209 DynamicVector<Request> mRequests; 210 211 //! The current report interval being sent to the session. This is only valid 212 //! if the mRequests is non-empty. 213 Milliseconds mCurrentInterval = Milliseconds(UINT64_MAX); 214 215 //! The state of the last successful request to the platform. 216 bool mPlatformEnabled = false; 217 218 //! True if a request from the CHRE framework is currently pending. 219 bool mInternalRequestPending = false; 220 221 //! True if a setting change event is pending to be processed. 222 bool mSettingChangePending = false; 223 224 //! True if a state resync callback is pending to be processed. 225 bool mResyncPending = false; 226 227 // Allows GnssManager to access constructor. 228 friend class GnssManager; 229 230 //! The histogram of error codes for collected errors, the index of this array 231 //! corresponds to the type of the errorcode 232 uint32_t mGnssErrorHistogram[CHRE_ERROR_SIZE] = {0}; 233 234 /** 235 * Constructs a GnssSesson. 236 * 237 * @param reportEventType The report event type of this GNSS session. 238 * Currently, only CHRE_EVENT_GNSS_LOCATION for a location session and 239 * CHRE_EVENT_GNSS_LOCATION for a measurement session are supported. 240 */ 241 GnssSession(uint16_t reportEventType); 242 243 /** 244 * Configures the GNSS engine to be enabled/disabled. If enable is set to true 245 * then the minInterval and minTimeToNext values are valid. 246 * 247 * @param nanoapp The nanoapp requesting the state change for the engine. 248 * @param enable Whether to enable or disable the engine. 249 * @param minInterval The minimum reporting interval requested by the nanoapp. 250 * @param minTimeToNext The minimum time to the next report. 251 * @param cookie The cookie provided by the nanoapp to round-trip for context. 252 * 253 * @return true if the request was accepted. 254 */ 255 bool configure(Nanoapp *nanoapp, bool enable, Milliseconds minInterval, 256 Milliseconds minTimeToNext, const void *cookie); 257 258 /** 259 * Checks if a nanoapp has an open session request. 260 * 261 * @param instanceId The nanoapp instance ID to search for. 262 * @param requestIndex A pointer to an index to populate if the nanoapp has an 263 * open session request. 264 * 265 * @return true if the provided instanceId was found. 266 */ 267 bool nanoappHasRequest(uint16_t instanceId, 268 size_t *requestIndex = nullptr) const; 269 270 /** 271 * Adds a request for a session to the queue of state transitions. 272 * 273 * @param instanceId The nanoapp instance ID requesting a session. 274 * @param enable Whether the session is being enabled or disabled for this 275 * nanoapp. 276 * @param minInterval The minimum interval requested by the nanoapp. 277 * @param cookie A cookie that is round-tripped to the nanoapp for context. 278 * 279 * @return true if the state transition was added to the queue. 280 */ 281 bool addRequestToQueue(uint16_t instanceId, bool enable, 282 Milliseconds minInterval, const void *cookie); 283 284 /** 285 * @return true if the session is currently enabled. 286 */ 287 bool isEnabled() const; 288 289 /** 290 * Determines if a change to the session state is required given a set of 291 * parameters. 292 * 293 * @param requestedState The target state requested by a nanoapp. 294 * @param minInterval The minimum reporting interval. 295 * @param nanoappHasRequest If the nanoapp already has a request. 296 * @param requestIndex The index of the request in the list of open requests 297 * if nanoappHasRequest is set to true. 298 * 299 * @return true if a state transition is required. 300 */ 301 bool stateTransitionIsRequired(bool requestedState, Milliseconds minInterval, 302 bool nanoappHasRequest, 303 size_t requestIndex) const; 304 305 /** 306 * Updates the session requests given a nanoapp and the interval requested. 307 * 308 * @param enable true if enabling the session. 309 * @param minInterval the minimum reporting interval if enable is true. 310 * @param instanceId the nanoapp instance ID that owns the request. 311 * 312 * @return true if the session request list was updated. 313 */ 314 bool updateRequests(bool enable, Milliseconds minInterval, 315 uint16_t instanceId); 316 317 /** 318 * Posts the result of a GNSS session add/remove request. 319 * 320 * @param instanceId The nanoapp instance ID that made the request. 321 * @param success true if the operation was successful. 322 * @param enable true if enabling the session. 323 * @param minInterval the minimum reporting interval. 324 * @param errorCode the error code as a result of this operation. 325 * @param cookie the cookie that the nanoapp is provided for context. 326 * 327 * @return true if the event was successfully posted. 328 */ 329 bool postAsyncResultEvent(uint16_t instanceId, bool success, bool enable, 330 Milliseconds minInterval, uint8_t errorCode, 331 const void *cookie); 332 333 /** 334 * Calls through to postAsyncResultEvent but invokes FATAL_ERROR if the 335 * event is not posted successfully. This is used in asynchronous contexts 336 * where a nanoapp could be stuck waiting for a response but CHRE failed to 337 * enqueue one. For parameter details, 338 * @see postAsyncResultEvent 339 */ 340 void postAsyncResultEventFatal(uint16_t instanceId, bool success, bool enable, 341 Milliseconds minInterval, uint8_t errorCode, 342 const void *cookie); 343 344 /** 345 * Handles the result of a request to PlatformGnss to change the state of 346 * the session. See the handleStatusChange method which may be called from 347 * any thread. This method is intended to be invoked on the CHRE event loop 348 * thread. 349 * 350 * @param enabled true if the session was enabled 351 * @param errorCode an error code that is provided to indicate success. 352 */ 353 void handleStatusChangeSync(bool enabled, uint8_t errorCode); 354 355 /** 356 * Releases a GNSS report event after nanoapps have consumed it. 357 * 358 * @param eventType the type of event being freed. 359 * @param eventData a pointer to the scan event to release. 360 */ 361 static void freeReportEventCallback(uint16_t eventType, void *eventData); 362 363 /** 364 * Configures PlatformGnss based on session settings. 365 * 366 * @return true if PlatformGnss has accepted the setting. 367 */ 368 bool controlPlatform(bool enable, Milliseconds minInterval, 369 Milliseconds minTimeToNext); 370 371 /** 372 * Add a log to list of session logs possibly pushing out the oldest log. 373 * 374 * @param nanoappInstanceId the instance of id of nanoapp requesting 375 * @param interval the interval in milliseconds for request 376 * @param start true if the is a start request, false if a stop request 377 */ 378 void addSessionRequestLog(uint16_t nanoappInstanceId, Milliseconds interval, 379 bool start); 380 381 /** 382 * Dispatches pending state transitions on the queue until the first one 383 * succeeds. 384 */ 385 void dispatchQueuedStateTransitions(); 386 }; 387 388 /** 389 * The GnssManager handles platform init, capability query, and delagates debug 390 * dump and all GNSS request management to GnssSession(s), which includes 391 * multiplexing multiple requests into one for the platform to handle. 392 * 393 * This class is effectively a singleton as there can only be one instance of 394 * the PlatformGnss instance. 395 */ 396 class GnssManager : public NonCopyable { 397 public: 398 /** 399 * Constructs a GnssManager. 400 */ 401 GnssManager(); 402 403 /** 404 * Initializes the underlying platform-specific GNSS module. Must be called 405 * prior to invoking any other methods in this class. 406 */ 407 void init(); 408 409 /** 410 * @return the GNSS capabilities exposed by this platform. 411 */ 412 uint32_t getCapabilities(); 413 getLocationSession()414 GnssSession &getLocationSession() { 415 return mLocationSession; 416 }; 417 getMeasurementSession()418 GnssSession &getMeasurementSession() { 419 return mMeasurementSession; 420 }; 421 422 /** 423 * Invoked when the host notifies CHRE of a settings change. 424 * 425 * @param setting The setting that changed. 426 * @param enabled Whether setting is enabled or not. 427 */ 428 void onSettingChanged(Setting setting, bool enabled); 429 430 /** 431 * Invoked as a result of a requestStateResync() callback from the GNSS PAL. 432 * Runs asynchronously in the context of the callback immediately. 433 */ 434 void handleRequestStateResyncCallback(); 435 436 /** 437 * Invoked as a result of a requestStateResync() callback from the GNSS PAL. 438 * Runs in the context of the CHRE thread. 439 */ 440 void handleRequestStateResyncCallbackSync(); 441 442 /** 443 * @param nanoapp The nanoapp invoking 444 * chreGnssConfigurePassiveLocationListener. 445 * @param enable true to enable the configuration. 446 * 447 * @return true if the configuration succeeded. 448 */ 449 bool configurePassiveLocationListener(Nanoapp *nanoapp, bool enable); 450 451 /** 452 * Prints state in a string buffer. Must only be called from the context of 453 * the main CHRE thread. 454 * 455 * @param debugDump The debug dump wrapper where a string can be printed 456 * into one of the buffers. 457 */ 458 void logStateToBuffer(DebugDumpWrapper &debugDump) const; 459 460 /** 461 * Disables the location session, the measurement session and the passive 462 * location listener associated to a nanoapp. 463 * 464 * @param nanoapp A non-null pointer to the nanoapp. 465 * 466 * @return The number of subscriptions disabled. 467 */ 468 uint32_t disableAllSubscriptions(Nanoapp *nanoapp); 469 470 private: 471 // Allows GnssSession to access mPlatformGnss. 472 friend class GnssSession; 473 474 //! The platform GNSS interface. 475 PlatformGnss mPlatformGnss; 476 477 //! The instance of location session. 478 GnssSession mLocationSession; 479 480 //! The instance of measurement session. 481 GnssSession mMeasurementSession; 482 483 //! The list of instance ID of nanoapps that has a passive location listener 484 //! request. 485 DynamicVector<uint16_t> mPassiveLocationListenerNanoapps; 486 487 //! true if the passive location listener is enabled at the platform. 488 bool mPlatformPassiveLocationListenerEnabled; 489 490 /** 491 * @param nanoappInstanceId The instance ID of the nanoapp to check. 492 * @param index If non-null and this function returns true, stores the index 493 * of mPassiveLocationListenerNanoapps where the instance ID is stored. 494 * 495 * @return true if the nanoapp currently has a passive location listener 496 * request. 497 */ 498 bool nanoappHasPassiveLocationListener(uint16_t nanoappInstanceId, 499 size_t *index = nullptr); 500 501 /** 502 * Helper function to invoke configurePassiveLocationListener at the platform 503 * and handle the result. 504 * 505 * @param enable true to enable the configuration. 506 * 507 * @return true if success. 508 */ 509 bool platformConfigurePassiveLocationListener(bool enable); 510 }; 511 512 } // namespace chre 513 514 #endif // CHRE_CORE_GNSS_MANAGER_H_ 515