/* * Copyright (C) 2020 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "GnssAidl" #include "Gnss.h" #include #include #include #include "AGnss.h" #include "AGnssRil.h" #include "DeviceFileReader.h" #include "FixLocationParser.h" #include "GnssAntennaInfo.h" #include "GnssBatching.h" #include "GnssConfiguration.h" #include "GnssDebug.h" #include "GnssGeofence.h" #include "GnssNavigationMessageInterface.h" #include "GnssPsds.h" #include "GnssVisibilityControl.h" #include "MeasurementCorrectionsInterface.h" #include "Utils.h" namespace aidl::android::hardware::gnss { using ::android::hardware::gnss::common::Utils; using ndk::ScopedAStatus; using GnssSvInfo = IGnssCallback::GnssSvInfo; constexpr int TTFF_MILLIS = 2200; std::shared_ptr Gnss::sGnssCallback = nullptr; Gnss::Gnss() : mMinIntervalMs(1000), mFirstFixReceived(false) {} ScopedAStatus Gnss::setCallback(const std::shared_ptr& callback) { ALOGD("setCallback"); if (callback == nullptr) { ALOGE("%s: Null callback ignored", __func__); return ScopedAStatus::fromExceptionCode(STATUS_INVALID_OPERATION); } sGnssCallback = callback; int capabilities = (int)(IGnssCallback::CAPABILITY_MEASUREMENTS | IGnssCallback::CAPABILITY_SCHEDULING | IGnssCallback::CAPABILITY_SATELLITE_BLOCKLIST | IGnssCallback::CAPABILITY_SATELLITE_PVT | IGnssCallback::CAPABILITY_CORRELATION_VECTOR | IGnssCallback::CAPABILITY_ANTENNA_INFO); auto status = sGnssCallback->gnssSetCapabilitiesCb(capabilities); if (!status.isOk()) { ALOGE("%s: Unable to invoke callback.gnssSetCapabilitiesCb", __func__); } IGnssCallback::GnssSystemInfo systemInfo = { .yearOfHw = 2022, .name = "Google, Cuttlefish, AIDL v2", }; status = sGnssCallback->gnssSetSystemInfoCb(systemInfo); if (!status.isOk()) { ALOGE("%s: Unable to invoke callback.gnssSetSystemInfoCb", __func__); } return ScopedAStatus::ok(); } std::unique_ptr Gnss::getLocationFromHW() { if (!::android::hardware::gnss::common::ReplayUtils::hasFixedLocationDeviceFile()) { return nullptr; } std::string inputStr = ::android::hardware::gnss::common::DeviceFileReader::Instance().getLocationData(); return ::android::hardware::gnss::common::FixLocationParser::getLocationFromInputStr(inputStr); } ScopedAStatus Gnss::start() { ALOGD("start()"); if (mIsActive) { ALOGW("Gnss has started. Restarting..."); stop(); } mIsActive = true; mThreadBlocker.reset(); // notify measurement engine to update measurement interval mGnssMeasurementInterface->setLocationEnabled(true); this->reportGnssStatusValue(IGnssCallback::GnssStatusValue::SESSION_BEGIN); mThread = std::thread([this]() { this->reportSvStatus(); if (!mFirstFixReceived) { std::this_thread::sleep_for(std::chrono::milliseconds(TTFF_MILLIS)); mFirstFixReceived = true; } do { if (!mIsActive) { break; } this->reportSvStatus(); this->reportNmea(); auto currentLocation = getLocationFromHW(); mGnssPowerIndication->notePowerConsumption(); if (currentLocation != nullptr) { this->reportLocation(*currentLocation); } else { const auto location = Utils::getMockLocation(); this->reportLocation(location); } } while (mIsActive && mThreadBlocker.wait_for(std::chrono::milliseconds(mMinIntervalMs))); }); return ScopedAStatus::ok(); } ScopedAStatus Gnss::stop() { ALOGD("stop"); mIsActive = false; mGnssMeasurementInterface->setLocationEnabled(false); this->reportGnssStatusValue(IGnssCallback::GnssStatusValue::SESSION_END); mThreadBlocker.notify(); if (mThread.joinable()) { mThread.join(); } return ScopedAStatus::ok(); } ScopedAStatus Gnss::close() { ALOGD("close"); sGnssCallback = nullptr; return ScopedAStatus::ok(); } void Gnss::reportLocation(const GnssLocation& location) const { std::unique_lock lock(mMutex); if (sGnssCallback == nullptr) { ALOGE("%s: GnssCallback is null.", __func__); return; } auto status = sGnssCallback->gnssLocationCb(location); if (!status.isOk()) { ALOGE("%s: Unable to invoke gnssLocationCb", __func__); } return; } void Gnss::reportSvStatus() const { if (mIsSvStatusActive) { auto svStatus = filterBlocklistedSatellites(Utils::getMockSvInfoList()); reportSvStatus(svStatus); } } void Gnss::reportSvStatus(const std::vector& svInfoList) const { std::unique_lock lock(mMutex); if (sGnssCallback == nullptr) { ALOGE("%s: sGnssCallback is null.", __func__); return; } auto status = sGnssCallback->gnssSvStatusCb(svInfoList); if (!status.isOk()) { ALOGE("%s: Unable to invoke callback", __func__); } } std::vector Gnss::filterBlocklistedSatellites( std::vector gnssSvInfoList) const { for (uint32_t i = 0; i < gnssSvInfoList.size(); i++) { if (mGnssConfiguration->isBlocklisted(gnssSvInfoList[i])) { gnssSvInfoList[i].svFlag &= ~(uint32_t)IGnssCallback::GnssSvFlags::USED_IN_FIX; } } return gnssSvInfoList; } void Gnss::reportGnssStatusValue(const IGnssCallback::GnssStatusValue gnssStatusValue) const { std::unique_lock lock(mMutex); if (sGnssCallback == nullptr) { ALOGE("%s: sGnssCallback is null.", __func__); return; } auto status = sGnssCallback->gnssStatusCb(gnssStatusValue); if (!status.isOk()) { ALOGE("%s: Unable to invoke gnssStatusCb", __func__); } } void Gnss::reportNmea() const { if (mIsNmeaActive) { std::unique_lock lock(mMutex); if (sGnssCallback == nullptr) { ALOGE("%s: sGnssCallback is null.", __func__); return; } nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC); auto status = sGnssCallback->gnssNmeaCb(now, "$TEST,0,1,2,3,4,5"); if (!status.isOk()) { ALOGE("%s: Unable to invoke callback", __func__); } } } ScopedAStatus Gnss::startSvStatus() { ALOGD("startSvStatus"); mIsSvStatusActive = true; return ScopedAStatus::ok(); } ScopedAStatus Gnss::stopSvStatus() { ALOGD("stopSvStatus"); mIsSvStatusActive = false; return ScopedAStatus::ok(); } ScopedAStatus Gnss::startNmea() { ALOGD("startNmea"); mIsNmeaActive = true; return ScopedAStatus::ok(); } ScopedAStatus Gnss::stopNmea() { ALOGD("stopNmea"); mIsNmeaActive = false; return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionAGnss(std::shared_ptr* iAGnss) { ALOGD("Gnss::getExtensionAGnss"); *iAGnss = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } ScopedAStatus Gnss::injectTime(int64_t timeMs, int64_t timeReferenceMs, int uncertaintyMs) { ALOGD("injectTime. timeMs:%" PRId64 ", timeReferenceMs:%" PRId64 ", uncertaintyMs:%d", timeMs, timeReferenceMs, uncertaintyMs); return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionAGnssRil(std::shared_ptr* iAGnssRil) { ALOGD("Gnss::getExtensionAGnssRil"); *iAGnssRil = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } ScopedAStatus Gnss::injectLocation(const GnssLocation& location) { ALOGD("injectLocation. lat:%lf, lng:%lf, acc:%f", location.latitudeDegrees, location.longitudeDegrees, location.horizontalAccuracyMeters); return ScopedAStatus::ok(); } ScopedAStatus Gnss::injectBestLocation(const GnssLocation& location) { ALOGD("injectBestLocation. lat:%lf, lng:%lf, acc:%f", location.latitudeDegrees, location.longitudeDegrees, location.horizontalAccuracyMeters); return ScopedAStatus::ok(); } ScopedAStatus Gnss::deleteAidingData(GnssAidingData aidingDataFlags) { ALOGD("deleteAidingData. flags:%d", (int)aidingDataFlags); mFirstFixReceived = false; return ScopedAStatus::ok(); } ScopedAStatus Gnss::setPositionMode(const PositionModeOptions& options) { ALOGD("setPositionMode. minIntervalMs:%d, lowPowerMode:%d", options.minIntervalMs, (int)options.lowPowerMode); mMinIntervalMs = std::max(1000, options.minIntervalMs); mGnssMeasurementInterface->setLocationInterval(mMinIntervalMs); return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionPsds(std::shared_ptr* iGnssPsds) { ALOGD("getExtensionPsds"); *iGnssPsds = SharedRefBase::make(); return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssConfiguration( std::shared_ptr* iGnssConfiguration) { ALOGD("getExtensionGnssConfiguration"); if (mGnssConfiguration == nullptr) { mGnssConfiguration = SharedRefBase::make(); } *iGnssConfiguration = mGnssConfiguration; return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssPowerIndication( std::shared_ptr* iGnssPowerIndication) { ALOGD("getExtensionGnssPowerIndication"); if (mGnssPowerIndication == nullptr) { mGnssPowerIndication = SharedRefBase::make(); } *iGnssPowerIndication = mGnssPowerIndication; return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssMeasurement( std::shared_ptr* iGnssMeasurement) { ALOGD("getExtensionGnssMeasurement"); if (mGnssMeasurementInterface == nullptr) { mGnssMeasurementInterface = SharedRefBase::make(); } *iGnssMeasurement = mGnssMeasurementInterface; return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssBatching(std::shared_ptr* iGnssBatching) { ALOGD("getExtensionGnssBatching"); *iGnssBatching = SharedRefBase::make(); return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssGeofence(std::shared_ptr* iGnssGeofence) { ALOGD("getExtensionGnssGeofence"); *iGnssGeofence = SharedRefBase::make(); return ScopedAStatus::ok(); } ScopedAStatus Gnss::getExtensionGnssNavigationMessage( std::shared_ptr* iGnssNavigationMessage) { ALOGD("getExtensionGnssNavigationMessage"); *iGnssNavigationMessage = SharedRefBase::make(); return ScopedAStatus::ok(); } ndk::ScopedAStatus Gnss::getExtensionGnssDebug(std::shared_ptr* iGnssDebug) { ALOGD("Gnss::getExtensionGnssDebug"); *iGnssDebug = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } ndk::ScopedAStatus Gnss::getExtensionGnssVisibilityControl( std::shared_ptr* iGnssVisibilityControl) { ALOGD("Gnss::getExtensionGnssVisibilityControl"); *iGnssVisibilityControl = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } ndk::ScopedAStatus Gnss::getExtensionGnssAntennaInfo( std::shared_ptr* iGnssAntennaInfo) { ALOGD("Gnss::getExtensionGnssAntennaInfo"); *iGnssAntennaInfo = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } ndk::ScopedAStatus Gnss::getExtensionMeasurementCorrections( std::shared_ptr* iMeasurementCorrections) { ALOGD("Gnss::getExtensionMeasurementCorrections"); *iMeasurementCorrections = SharedRefBase::make(); return ndk::ScopedAStatus::ok(); } } // namespace aidl::android::hardware::gnss