You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
209 lines
6.8 KiB
209 lines
6.8 KiB
/*
|
|
* 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.
|
|
*/
|
|
|
|
#include "chre/core/nanoapp.h"
|
|
|
|
#include "chre/core/event_loop_manager.h"
|
|
#include "chre/platform/assert.h"
|
|
#include "chre/platform/fatal_error.h"
|
|
#include "chre/platform/log.h"
|
|
#include "chre/util/system/debug_dump.h"
|
|
#include "chre_api/chre/gnss.h"
|
|
#include "chre_api/chre/version.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#if CHRE_FIRST_SUPPORTED_API_VERSION < CHRE_API_VERSION_1_5
|
|
#define CHRE_GNSS_MEASUREMENT_BACK_COMPAT_ENABLED
|
|
#endif
|
|
|
|
namespace chre {
|
|
|
|
constexpr size_t Nanoapp::kMaxSizeWakeupBuckets;
|
|
|
|
Nanoapp::Nanoapp() {
|
|
// Push first bucket onto wakeup bucket queue
|
|
cycleWakeupBuckets(1);
|
|
}
|
|
|
|
Nanoapp::~Nanoapp() {
|
|
const size_t totalAllocatedBytes = getTotalAllocatedBytes();
|
|
|
|
if (totalAllocatedBytes > 0) {
|
|
// TODO: Consider asserting here
|
|
LOGE("Nanoapp ID=0x%016" PRIx64 " still has %zu allocated bytes!",
|
|
getAppId(), totalAllocatedBytes);
|
|
}
|
|
}
|
|
|
|
bool Nanoapp::isRegisteredForBroadcastEvent(uint16_t eventType,
|
|
uint16_t targetGroupIdMask) const {
|
|
bool registered = false;
|
|
size_t foundIndex = registrationIndex(eventType);
|
|
if (foundIndex < mRegisteredEvents.size()) {
|
|
const EventRegistration ® = mRegisteredEvents[foundIndex];
|
|
if (targetGroupIdMask & reg.groupIdMask) {
|
|
registered = true;
|
|
}
|
|
}
|
|
return registered;
|
|
}
|
|
|
|
void Nanoapp::registerForBroadcastEvent(uint16_t eventType,
|
|
uint16_t groupIdMask) {
|
|
size_t foundIndex = registrationIndex(eventType);
|
|
if (foundIndex < mRegisteredEvents.size()) {
|
|
mRegisteredEvents[foundIndex].groupIdMask |= groupIdMask;
|
|
} else if (!mRegisteredEvents.push_back(
|
|
EventRegistration(eventType, groupIdMask))) {
|
|
FATAL_ERROR_OOM();
|
|
}
|
|
}
|
|
|
|
void Nanoapp::unregisterForBroadcastEvent(uint16_t eventType,
|
|
uint16_t groupIdMask) {
|
|
size_t foundIndex = registrationIndex(eventType);
|
|
if (foundIndex < mRegisteredEvents.size()) {
|
|
EventRegistration ® = mRegisteredEvents[foundIndex];
|
|
reg.groupIdMask &= ~groupIdMask;
|
|
if (reg.groupIdMask == 0) {
|
|
mRegisteredEvents.erase(foundIndex);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Nanoapp::configureNanoappInfoEvents(bool enable) {
|
|
if (enable) {
|
|
registerForBroadcastEvent(CHRE_EVENT_NANOAPP_STARTED);
|
|
registerForBroadcastEvent(CHRE_EVENT_NANOAPP_STOPPED);
|
|
} else {
|
|
unregisterForBroadcastEvent(CHRE_EVENT_NANOAPP_STARTED);
|
|
unregisterForBroadcastEvent(CHRE_EVENT_NANOAPP_STOPPED);
|
|
}
|
|
}
|
|
|
|
void Nanoapp::configureHostSleepEvents(bool enable) {
|
|
if (enable) {
|
|
registerForBroadcastEvent(CHRE_EVENT_HOST_AWAKE);
|
|
registerForBroadcastEvent(CHRE_EVENT_HOST_ASLEEP);
|
|
} else {
|
|
unregisterForBroadcastEvent(CHRE_EVENT_HOST_AWAKE);
|
|
unregisterForBroadcastEvent(CHRE_EVENT_HOST_ASLEEP);
|
|
}
|
|
}
|
|
|
|
void Nanoapp::configureDebugDumpEvent(bool enable) {
|
|
if (enable) {
|
|
registerForBroadcastEvent(CHRE_EVENT_DEBUG_DUMP);
|
|
} else {
|
|
unregisterForBroadcastEvent(CHRE_EVENT_DEBUG_DUMP);
|
|
}
|
|
}
|
|
|
|
void Nanoapp::configureUserSettingEvent(uint8_t setting, bool enable) {
|
|
if (enable) {
|
|
registerForBroadcastEvent(CHRE_EVENT_SETTING_CHANGED_FIRST_EVENT + setting);
|
|
} else {
|
|
unregisterForBroadcastEvent(CHRE_EVENT_SETTING_CHANGED_FIRST_EVENT +
|
|
setting);
|
|
}
|
|
}
|
|
|
|
Event *Nanoapp::processNextEvent() {
|
|
Event *event = mEventQueue.pop();
|
|
|
|
CHRE_ASSERT_LOG(event != nullptr, "Tried delivering event, but queue empty");
|
|
if (event != nullptr) {
|
|
if (event->eventType == CHRE_EVENT_GNSS_DATA) {
|
|
handleGnssMeasurementDataEvent(event);
|
|
} else {
|
|
handleEvent(event->senderInstanceId, event->eventType, event->eventData);
|
|
}
|
|
}
|
|
|
|
return event;
|
|
}
|
|
|
|
void Nanoapp::blameHostWakeup() {
|
|
if (mWakeupBuckets.back() < UINT16_MAX) ++mWakeupBuckets.back();
|
|
}
|
|
|
|
void Nanoapp::cycleWakeupBuckets(size_t numBuckets) {
|
|
numBuckets = std::min(numBuckets, kMaxSizeWakeupBuckets);
|
|
for (size_t i = 0; i < numBuckets; ++i) {
|
|
if (mWakeupBuckets.full()) {
|
|
mWakeupBuckets.erase(0);
|
|
}
|
|
mWakeupBuckets.push_back(0);
|
|
}
|
|
}
|
|
|
|
void Nanoapp::logStateToBuffer(DebugDumpWrapper &debugDump) const {
|
|
debugDump.print(" Id=%" PRIu32 " 0x%016" PRIx64 " ", getInstanceId(),
|
|
getAppId());
|
|
PlatformNanoapp::logStateToBuffer(debugDump);
|
|
debugDump.print(" v%" PRIu32 ".%" PRIu32 ".%" PRIu32 " tgtAPI=%" PRIu32
|
|
".%" PRIu32 " curAlloc=%zu peakAlloc=%zu",
|
|
CHRE_EXTRACT_MAJOR_VERSION(getAppVersion()),
|
|
CHRE_EXTRACT_MINOR_VERSION(getAppVersion()),
|
|
CHRE_EXTRACT_PATCH_VERSION(getAppVersion()),
|
|
CHRE_EXTRACT_MAJOR_VERSION(getTargetApiVersion()),
|
|
CHRE_EXTRACT_MINOR_VERSION(getTargetApiVersion()),
|
|
getTotalAllocatedBytes(), getPeakAllocatedBytes());
|
|
debugDump.print(" hostWakeups=[ cur->");
|
|
// Get buckets latest -> earliest except last one
|
|
for (size_t i = mWakeupBuckets.size() - 1; i > 0; --i) {
|
|
debugDump.print("%" PRIu16 ", ", mWakeupBuckets[i]);
|
|
}
|
|
// Earliest bucket gets no comma
|
|
debugDump.print("%" PRIu16 " ]\n", mWakeupBuckets.front());
|
|
}
|
|
|
|
bool Nanoapp::permitPermissionUse(uint32_t permission) const {
|
|
return !supportsAppPermissions() ||
|
|
((getAppPermissions() & permission) == permission);
|
|
}
|
|
|
|
size_t Nanoapp::registrationIndex(uint16_t eventType) const {
|
|
size_t foundIndex = 0;
|
|
for (; foundIndex < mRegisteredEvents.size(); ++foundIndex) {
|
|
const EventRegistration ® = mRegisteredEvents[foundIndex];
|
|
if (reg.eventType == eventType) {
|
|
break;
|
|
}
|
|
}
|
|
return foundIndex;
|
|
}
|
|
|
|
void Nanoapp::handleGnssMeasurementDataEvent(const Event *event) {
|
|
#ifdef CHRE_GNSS_MEASUREMENT_BACK_COMPAT_ENABLED
|
|
const struct chreGnssDataEvent *data =
|
|
static_cast<const struct chreGnssDataEvent *>(event->eventData);
|
|
if (getTargetApiVersion() < CHRE_API_VERSION_1_5 &&
|
|
data->measurement_count > CHRE_GNSS_MAX_MEASUREMENT_PRE_1_5) {
|
|
chreGnssDataEvent localEvent;
|
|
memcpy(&localEvent, data, sizeof(struct chreGnssDataEvent));
|
|
localEvent.measurement_count = CHRE_GNSS_MAX_MEASUREMENT_PRE_1_5;
|
|
handleEvent(event->senderInstanceId, event->eventType, &localEvent);
|
|
} else
|
|
#endif // CHRE_GNSS_MEASUREMENT_BACK_COMPAT_ENABLED
|
|
{
|
|
handleEvent(event->senderInstanceId, event->eventType, event->eventData);
|
|
}
|
|
}
|
|
|
|
} // namespace chre
|