225 lines
7.6 KiB
C++
225 lines
7.6 KiB
C++
/*
|
|
* Copyright (C) 2016 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 "GnssHAL_GnssBatchingInterface"
|
|
|
|
#include "GnssBatching.h"
|
|
#include <Gnss.h> // for wakelock consolidation
|
|
#include <GnssUtils.h>
|
|
|
|
#include <cutils/log.h> // for ALOGE
|
|
#include <vector>
|
|
|
|
namespace android {
|
|
namespace hardware {
|
|
namespace gnss {
|
|
namespace V1_0 {
|
|
namespace implementation {
|
|
|
|
sp<IGnssBatchingCallback> GnssBatching::sGnssBatchingCbIface = nullptr;
|
|
bool GnssBatching::sFlpSupportsBatching = false;
|
|
|
|
FlpCallbacks GnssBatching::sFlpCb = {
|
|
.size = sizeof(FlpCallbacks),
|
|
.location_cb = locationCb,
|
|
.acquire_wakelock_cb = acquireWakelockCb,
|
|
.release_wakelock_cb = releaseWakelockCb,
|
|
.set_thread_event_cb = setThreadEventCb,
|
|
.flp_capabilities_cb = flpCapabilitiesCb,
|
|
.flp_status_cb = flpStatusCb,
|
|
};
|
|
|
|
GnssBatching::GnssBatching(const FlpLocationInterface* flpLocationIface) :
|
|
mFlpLocationIface(flpLocationIface) {
|
|
}
|
|
|
|
/*
|
|
* This enum is used locally by various methods below. It is only used by the default
|
|
* implementation and is not part of the GNSS interface.
|
|
*/
|
|
enum BatchingValues : uint16_t {
|
|
// Numbers 0-3 were used in earlier implementations - using 4 to be distinct to the HAL
|
|
FLP_GNSS_BATCHING_CLIENT_ID = 4,
|
|
// Tech. mask of GNSS, and sensor aiding, for legacy HAL to fit with GnssBatching API
|
|
FLP_TECH_MASK_GNSS_AND_SENSORS = FLP_TECH_MASK_GNSS | FLP_TECH_MASK_SENSORS,
|
|
// Putting a cap to avoid possible memory issues. Unlikely values this high are supported.
|
|
MAX_LOCATIONS_PER_BATCH = 1000
|
|
};
|
|
|
|
void GnssBatching::locationCb(int32_t locationsCount, FlpLocation** locations) {
|
|
if (sGnssBatchingCbIface == nullptr) {
|
|
ALOGE("%s: GNSS Batching Callback Interface configured incorrectly", __func__);
|
|
return;
|
|
}
|
|
|
|
if (locations == nullptr) {
|
|
ALOGE("%s: Invalid locations from GNSS HAL", __func__);
|
|
return;
|
|
}
|
|
|
|
if (locationsCount < 0) {
|
|
ALOGE("%s: Negative location count: %d set to 0", __func__, locationsCount);
|
|
locationsCount = 0;
|
|
} else if (locationsCount > MAX_LOCATIONS_PER_BATCH) {
|
|
ALOGW("%s: Unexpected high location count: %d set to %d", __func__, locationsCount,
|
|
MAX_LOCATIONS_PER_BATCH);
|
|
locationsCount = MAX_LOCATIONS_PER_BATCH;
|
|
}
|
|
|
|
/**
|
|
* Note:
|
|
* Some existing implementations may drop duplicate locations. These could be expanded here
|
|
* but as there's ambiguity between no-GPS-fix vs. dropped duplicates in that implementation,
|
|
* and that's not specified by the fused_location.h, that isn't safe to do here.
|
|
* Fortunately, this shouldn't be a major issue in cases where GNSS batching is typically
|
|
* used (e.g. when user is likely in vehicle/bicycle.)
|
|
*/
|
|
std::vector<android::hardware::gnss::V1_0::GnssLocation> gnssLocations;
|
|
for (int iLocation = 0; iLocation < locationsCount; iLocation++) {
|
|
if (locations[iLocation] == nullptr) {
|
|
ALOGE("%s: Null location at slot: %d of %d, skipping", __func__, iLocation,
|
|
locationsCount);
|
|
continue;
|
|
}
|
|
if ((locations[iLocation]->sources_used & ~FLP_TECH_MASK_GNSS_AND_SENSORS) != 0)
|
|
{
|
|
ALOGE("%s: Unrequested location type %d at slot: %d of %d, skipping", __func__,
|
|
locations[iLocation]->sources_used, iLocation, locationsCount);
|
|
continue;
|
|
}
|
|
gnssLocations.push_back(convertToGnssLocation(locations[iLocation]));
|
|
}
|
|
|
|
auto ret = sGnssBatchingCbIface->gnssLocationBatchCb(gnssLocations);
|
|
if (!ret.isOk()) {
|
|
ALOGE("%s: Unable to invoke callback", __func__);
|
|
}
|
|
}
|
|
|
|
void GnssBatching::acquireWakelockCb() {
|
|
Gnss::acquireWakelockFused();
|
|
}
|
|
|
|
void GnssBatching::releaseWakelockCb() {
|
|
Gnss::releaseWakelockFused();
|
|
}
|
|
|
|
// this can just return success, because threads are now set up on demand in the jni layer
|
|
int32_t GnssBatching::setThreadEventCb(ThreadEvent /*event*/) {
|
|
return FLP_RESULT_SUCCESS;
|
|
}
|
|
|
|
void GnssBatching::flpCapabilitiesCb(int32_t capabilities) {
|
|
ALOGD("%s capabilities %d", __func__, capabilities);
|
|
|
|
if (capabilities & CAPABILITY_GNSS) {
|
|
// once callback is received and capabilities high enough, we know version is
|
|
// high enough for flush()
|
|
sFlpSupportsBatching = true;
|
|
}
|
|
}
|
|
|
|
void GnssBatching::flpStatusCb(int32_t status) {
|
|
ALOGD("%s (default implementation) not forwarding status: %d", __func__, status);
|
|
}
|
|
|
|
// Methods from ::android::hardware::gnss::V1_0::IGnssBatching follow.
|
|
Return<bool> GnssBatching::init(const sp<IGnssBatchingCallback>& callback) {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching is unavailable", __func__);
|
|
return false;
|
|
}
|
|
|
|
sGnssBatchingCbIface = callback;
|
|
|
|
return (mFlpLocationIface->init(&sFlpCb) == 0);
|
|
}
|
|
|
|
Return<uint16_t> GnssBatching::getBatchSize() {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching interface is unavailable", __func__);
|
|
return 0;
|
|
}
|
|
|
|
return mFlpLocationIface->get_batch_size();
|
|
}
|
|
|
|
Return<bool> GnssBatching::start(const IGnssBatching::Options& options) {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching interface is unavailable", __func__);
|
|
return false;
|
|
}
|
|
|
|
if (!sFlpSupportsBatching) {
|
|
ALOGE("%s: Flp batching interface not supported, no capabilities callback received",
|
|
__func__);
|
|
return false;
|
|
}
|
|
|
|
FlpBatchOptions optionsHw;
|
|
// Legacy code used 9999 mW for High accuracy, and 21 mW for balanced.
|
|
// New GNSS API just expects reasonable GNSS chipset behavior - do something efficient
|
|
// given the interval. This 100 mW limit should be quite sufficient (esp. given legacy code
|
|
// implementations may not even use this value.)
|
|
optionsHw.max_power_allocation_mW = 100;
|
|
optionsHw.sources_to_use = FLP_TECH_MASK_GNSS_AND_SENSORS;
|
|
optionsHw.flags = 0;
|
|
if (options.flags & Flag::WAKEUP_ON_FIFO_FULL) {
|
|
optionsHw.flags |= FLP_BATCH_WAKEUP_ON_FIFO_FULL;
|
|
}
|
|
optionsHw.period_ns = options.periodNanos;
|
|
optionsHw.smallest_displacement_meters = 0; // Zero offset - just use time interval
|
|
|
|
return (mFlpLocationIface->start_batching(FLP_GNSS_BATCHING_CLIENT_ID, &optionsHw)
|
|
== FLP_RESULT_SUCCESS);
|
|
}
|
|
|
|
Return<void> GnssBatching::flush() {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching interface is unavailable", __func__);
|
|
return Void();
|
|
}
|
|
|
|
mFlpLocationIface->flush_batched_locations();
|
|
|
|
return Void();
|
|
}
|
|
|
|
Return<bool> GnssBatching::stop() {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching interface is unavailable", __func__);
|
|
return false;
|
|
}
|
|
|
|
return (mFlpLocationIface->stop_batching(FLP_GNSS_BATCHING_CLIENT_ID) == FLP_RESULT_SUCCESS);
|
|
}
|
|
|
|
Return<void> GnssBatching::cleanup() {
|
|
if (mFlpLocationIface == nullptr) {
|
|
ALOGE("%s: Flp batching interface is unavailable", __func__);
|
|
return Void();
|
|
}
|
|
|
|
mFlpLocationIface->cleanup();
|
|
|
|
return Void();
|
|
}
|
|
|
|
} // namespace implementation
|
|
} // namespace V1_0
|
|
} // namespace gnss
|
|
} // namespace hardware
|
|
} // namespace android
|